推荐学java——SpringMVC第一课 - 图1

SpringMVC 概述

Spring MVC 是 Spring 框架中的一个模块,也是 Spring 的核心之一,常用作 web 开发,其底层是 Servlet,故也叫 Spring web mvc。因为内内部架构模式是 mvc 架构,故此称为 Spring MVC .

SpringMVC 是一个容器,管理界面层中的控制器对象,使用 ioC 技术。

SpringMVC 的核心 Servlet ——DispatcherServlet,负责处理请求以及响应处理结果,这是我们学习的核心内容。

第一个 SpringMVC 例子

实现的功能(即需求)

用户发起请求,处理请求,给出响应结果。

实现步骤分析

  1. 创建web项目
  2. 添加依赖(spring-webmvc、servlet)
  3. 声明 SpringMVC 核心对象 DispatcherServlet
  4. 创建jsp,发起请求
  5. 创建一个普通类,作为控制器使用(代替之前的 servlet),这里要使用注解 @Controller @RequestMapping
  6. 创建展示结果的 jsp 页面
  7. 创建 SpringMVC 的配置文件(和 Spring 配置文件类似)

以上步骤中,第一步不需要多说;第二步中用到的依赖项如下:

  1. <!-- springMVC 依赖 -->
  2. <dependency>
  3. <groupId>org.springframework</groupId>
  4. <artifactId>spring-webmvc</artifactId>
  5. <version>5.3.15</version>
  6. </dependency>
  7. <!-- servlet依赖 -->
  8. <dependency>
  9. <groupId>javax.servlet</groupId>
  10. <artifactId>javax.servlet-api</artifactId>
  11. <version>4.0.1</version>
  12. </dependency>

声明 SpringMVC 核心对象 DispatcherServlet

首先在 resources 文件夹下创建 springmvc-servlet.xml,这是 springMVC 的配置文件,但现在我们只是创建文件,里面的配置后面会添加;然后在 web.xml 文件中添加如下代码:

  1. <!DOCTYPE web-app PUBLIC
  2. "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
  3. "http://java.sun.com/dtd/web-app_2_3.dtd" >
  4. <web-app>
  5. <servlet>
  6. <servlet-name>springmvc</servlet-name>
  7. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  8. <!-- 自定义SpringMVC的配置文件位置 -->
  9. <init-param>
  10. <param-name>contextConfigLocation</param-name>
  11. <param-value>classpath:springmvc-servlet.xml</param-value>
  12. </init-param>
  13. <!-- 表示服务器 tomcat 创建对象的顺序,数值越小,创建对象的时机越早,大于等于0的整数 -->
  14. <load-on-startup>1</load-on-startup>
  15. </servlet>
  16. <servlet-mapping>
  17. <servlet-name>springmvc</servlet-name>
  18. <!--
  19. <url-pattern> 把一些请求交给指定的 Servlet 处理。
  20. 那么就可以使用通配符来指定,常见的扩展名形式如:.do、.action、.mvc等
  21. -->
  22. <url-pattern>*.do</url-pattern>
  23. </servlet-mapping>
  24. </web-app>

这里的配置在上一节《Spring与web》中我们写过,这里略作调整,因为这次使用的是 DispatcherServlet ,同样,以上代码基本是固定的,记下来后面直接用即可。

创建发起请求的 jsp

页面很简单,做一个超链接,点击即可发起请求,具体代码如下:

  1. <%--
  2. Created by studyingJava
  3. Date: 2022/2/9
  4. Time: 11:57
  5. --%>
  6. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  7. <html>
  8. <head>
  9. <title>springmvc-first</title>
  10. </head>
  11. <body>
  12. <a href="hello.do">发起 hello.do 请求</a>
  13. </body>
  14. </html>

创建控制器类

新建一个名为 controller 的包,当然名字可以随你定义,完整代码如下:

  1. /**
  2. * desc: 自定义后端控制器,添加注解 @Controller,表示创建该java对象
  3. * author: 推荐学java
  4. * <p>
  5. * weChat: studyingJava
  6. */
  7. @Controller
  8. public class MyController {
  9. /**
  10. * @return ModelAndView:表示请求结果的封装(数据和视图)
  11. * @RequestMapping:请求映射,其参数value是一个字符串类型的数组,uri地址以 "/" 开头,method参数是指定请求方式的
  12. * 可用在方法上面和类上面。前者表示完整的地址,后者表示公共前缀,此时对应的方法上面只需要写前缀后面的部分即可
  13. */
  14. @RequestMapping(value = "/doHello")
  15. public ModelAndView doHello() {
  16. // 这里模拟结果 实际业务应该是调用service层
  17. ModelAndView modelAndView = new ModelAndView();
  18. modelAndView.addObject("msg", "处理 doHello 请求");
  19. modelAndView.addObject("code", 200);
  20. // 将请求结果给要展示的 jsp 页面
  21. modelAndView.setViewName("/result.jsp");
  22. return modelAndView;
  23. }
  24. }

创建展示结果的 jsp 页面

我们把控制器类中封装好的值取出来,显示在该页面上即可,代码如下:

  1. <%--
  2. Created by studyingJava
  3. Date: 2022/2/9
  4. Time: 16:43
  5. --%>
  6. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  7. <html>
  8. <head>
  9. <title>响应请求结果</title>
  10. </head>
  11. <body>
  12. /result.jsp 展示用户请求的处理结果<br>
  13. <h3>msg数据:${msg}</h3>
  14. <h3>code数据:${code}</h3>
  15. </body>
  16. </html>

在配置文件中添加注解扫描器

因为我们是通过注解来生成Java对象的,以及对应的请求响应方法也是通过注解指定的,所以程序要通过配置文件扫描我们给定的业务逻辑,代码如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
  6. <context:component-scan base-package="com.javafirst.controller"/>
  7. </beans>

测试结果

这里就需要用到我们上节的知识了,将我们的项目发布部署到本地 tomcat 服务器中,然后通过浏览器访问,看结果,具体操作这里不再重复,没有掌握的朋友可以看上一篇博客哈。

如果遇到 result.jsp 页面中的取值不显示,解决办法:修改 web.xml 中的 web-app 的开始标签为以下内容即可:

  1. <web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
  2. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
  4. http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">

最终效果如下,即为正确:

推荐学java——SpringMVC第一课 - 图2

扩展知识

以上内容中我们展示结果的页面写的是全路径,比较麻烦,那么系统给我们提供了一个叫视图解析器的配置,添加之后,我们只需要给页面名称即可,下面看具体配置。

在配置文件中添加如下代码:

  1. <!-- 视图解析器
  2. 只需要配置前缀和后缀即可,使用的时候只需要文件名
  3. -->
  4. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
  5. <property name="prefix" value="/WEB-INF/view/"/>
  6. <property name="suffix" value=".jsp"/>
  7. </bean>

MyController中的使用代码如下:

  1. // 将请求结果给要展示的 jsp 页面
  2. //modelAndView.setViewName("/WEB-INF/view/result.jsp");
  3. // 配置视图解析器后使用方式
  4. modelAndView.setViewName("result");

这样就ok了!

携带参数

上面我们的请求没有携带任何参数,但实际开发中很多情况需要携带参数,下面分情况演示。

携带多个参数

下面举个例子,在请求页面增加一个表单,用户输入两个数据,通过 Controller 接收后,显示到 result.jsp 页面上。

index.jsp增加如下代码:

  1. <body>
  2. <a href="hello.do">发起 hello.do 请求</a><br>
  3. <%--演示带参数请求--%>
  4. <form action="test/param.do" method="post">
  5. 语言:<input type="text" name="language"><br>
  6. 经验:<input type="text" name="work_time"><br>
  7. <input type="submit" value="提交">
  8. </form>
  9. </body>

MyController中我们新增一个方法来专门处理这个提交请求:

  1. /**
  2. * 传参-接收参数-显示参数
  3. * <p>
  4. * 形参类型建议使用基本数据类型的包装类,可以避免不填写造成的400异常
  5. *
  6. * @return
  7. */
  8. @RequestMapping(value = "/test/param.do")
  9. public ModelAndView doParam(String language, Integer work_time) {
  10. ModelAndView mv = new ModelAndView();
  11. mv.addObject("language", language);
  12. mv.addObject("work_time", work_time);
  13. mv.setViewName("result");
  14. return mv;
  15. }

结果各位自行验证,前面的跑通了,这个应该没啥问题,不懂的欢迎留言哈~

解决携带中文参数乱码问题

tips:如果我们传参使用中文,那么会发现显示的时候是乱码,那么我们需要通过过滤器来解决这个问题,代码如下:

web.xml中添加如下代码:

  1. <!-- 使用过滤器 解决乱码(请求过程、设置过程、显示过程)问题 -->
  2. <filter>
  3. <filter-name>characterEncodingFilter</filter-name>
  4. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  5. <init-param>
  6. <!-- 项目使用编码 -->
  7. <param-name>encoding</param-name>
  8. <param-value>utf-8</param-value>
  9. </init-param>
  10. <init-param>
  11. <!-- 强制请求使用encoding编码方式 -->
  12. <param-name>forceRequestEncoding</param-name>
  13. <param-value>true</param-value>
  14. </init-param>
  15. <init-param>
  16. <!-- 强制响应使用encoding编码方式 -->
  17. <param-name>forceResponseEncoding</param-name>
  18. <param-value>true</param-value>
  19. </init-param>
  20. </filter>
  21. <filter-mapping>
  22. <!-- 强制请求都走过滤器 -->
  23. <filter-name>characterEncodingFilter</filter-name>
  24. <url-pattern>/*</url-pattern>
  25. </filter-mapping>

增加过滤器就可以保证我们所有请求(请求和响应)都经过我们定义的过滤器,编码都是统一的,这段代码也是固定的,后续可直接使用。

请求中参数名和方法形参名不一致

tips:上面传参我们请求中的参数名和 Controller 中方法的形参名是一致的,那如果不一致,如何保证能收到请求中的参数值呢?

解决方案是通过注解 @RequestParam 注解,用法也很简单:

  1. @RequestMapping(value = "/test/param.do")
  2. public ModelAndView doParam1(@RequestParam(value = "language") String lan,
  3. @RequestParam(value = "work_time") Integer workTime) {
  4. ModelAndView mv = new ModelAndView();
  5. mv.addObject("language", lan);
  6. mv.addObject("work_time", workTime);
  7. mv.setViewName("result");
  8. return mv;
  9. }

该注解的 value 值是请求中的参数名,方法的形参就可以自定义了,该注解就会自动将请求中的参数值赋值给方法的形参。

控制器方法形参是 java 对象

这种接收参数方式有前提条件:

  • 请求中的参数名必须和 java 对象中的属性名保持一致
  • java对象需要提供无参构造方法

我们新建java对象 Programmer.java

  1. /**
  2. * desc:
  3. * author: 推荐学java
  4. * <p>
  5. * weChat: studyingJava
  6. */
  7. public class Programmer {
  8. private String languageType;
  9. private Integer workTime;
  10. public Programmer() {
  11. }
  12. public String getLanguageType() {
  13. return languageType;
  14. }
  15. public void setLanguageType(String languageType) {
  16. this.languageType = languageType;
  17. }
  18. public Integer getWorkTime() {
  19. return workTime;
  20. }
  21. public void setWorkTime(Integer workTime) {
  22. this.workTime = workTime;
  23. }
  24. }

index.jsp中新增一个表单:

  1. <%--演示 java对象接收 带参数的请求
  2. 要求:请求中的参数名和java对象中的属性名一致
  3. --%>
  4. <form action="test/param_object.do" method="post">
  5. 语言分类:<input type="text" name="languageType"><br>
  6. 工作经验:<input type="text" name="workTime"><br>
  7. <input type="submit" value="java对象接收请求参数">
  8. </form>

MyController.java中增加如下方法:

  1. /**
  2. * 演示 用java对象接收 带参数的请求
  3. * 要求:请求中的参数名和java对象中的属性名一致、java对象中提供无参构造方法
  4. *
  5. * @return
  6. */
  7. @RequestMapping(value = "/test/param_object.do")
  8. public ModelAndView doParamObject(Programmer programmer) {
  9. ModelAndView mv = new ModelAndView();
  10. mv.addObject("language", programmer.getLanguageType());
  11. mv.addObject("work_time", programmer.getWorkTime());
  12. mv.setViewName("result");
  13. return mv;
  14. }

结果各位读者朋友自行验证,这种接收参数的方式不支持请求中参数名和java对象中属性名不一致的情况。

控制器方法的返回值

包括这几种:

  • ModelAndView:数据和视图。这个我们已经学习过了。
  • String:只有视图,视图路径可以是完整路径,也可以是视图的逻辑名称。
  • void:
  • Object:

视图解析器返回 String 类型:
  1. /**
  2. * 视图控制器返回类型是String
  3. *
  4. * @return
  5. */
  6. @RequestMapping(value = "/test/controller_return_string.do")
  7. public String doReturnString(String languageType, Integer workTime) {
  8. // 使用视图解析器的逻辑名方式
  9. return "result";
  10. // 要使用全路径的话,那么不能使用视图解析器
  11. //return "/WEB-INF/view/result.jsp";
  12. }

如果只返回视图,还想附带数据,那么可以给该方法增加形参,如下方式:

  1. @RequestMapping(value = "/test/controller_return_string.do")
  2. public String doReturnString(HttpServletRequest request, String languageType, Integer workTime) {
  3. // 只返回视图 附带数据方式
  4. request.setAttribute("language", languageType);
  5. request.setAttribute("work_time", workTime);
  6. // 使用视图解析器的逻辑名方式
  7. return "result";
  8. }

视图解析器返回 void 类型:

完成AJAX请求。

第一步,在webapp下创建文件夹js

并将文件jquery-3.4.1.js放在该文件夹下。

第二步,添加依赖

既然是请求,那么对请求结果的数据做格式处理,通常是json格式,我们添加的就是json依赖:

  1. <!-- jackson依赖 -->
  2. <dependency>
  3. <groupId>com.fasterxml.jackson.core</groupId>
  4. <artifactId>jackson-core</artifactId>
  5. <version>2.9.10</version>
  6. </dependency>
  7. <dependency>
  8. <groupId>com.fasterxml.jackson.core</groupId>
  9. <artifactId>jackson-databind</artifactId>
  10. <version>2.9.10</version>
  11. </dependency>

依赖都是在pom.xml文件添加的。

第三步,在index.jsp页面中发起请求

首先在head标签下,引入我们添加的jquery-js库,代码如下:

  1. <script charset="UTF-8" type="text/javascript" src="<%=request.getContextPath()%>/js/jquery-3.4.1.js"></script>

这需要注意一点,可能会遇到这样的错误:

  1. Failed to load resource: the server responded with a status of 404

浏览器按F12就可以看到,是本地的 js文件 没有加载成功,那么大多数问题是以为路径的问题,所以这里的路径需要加上<%=request.getContextPath()%>,如果你一开始是这么写的 js/jquery-3.4.1.js,那么并不会出现语法报错,但会出现我写的这个错误,就是这样的原因,需要添加请求上下文路径。

然后在 body 中添加一个button,用于点击的时候发起ajax请求:

  1. <%--演示 视图控制器返回类型是void--%>
  2. <button id="btnAjax" type="button">发起Ajax请求</button>

然后在head标签里面,在我们刚刚写过的引入js代码下面写javaScript 脚本:

  1. <%-- 发起ajax请求 --%>
  2. <script type="text/javascript">
  3. $(function () {
  4. $("#btnAjax").on("click",function () {
  5. alert("click success")
  6. })
  7. })
  8. </script>

我们先测试一下点击后是否会生效,结果各位自己验证,是没有任何问题的,接下来我们就写具体的响应数据,也就是要包装json格式。

以上没有问题后,我们要演示Ajax真正的请求响应结果了,我们修改脚本如下:

  1. <%-- 发起ajax请求 --%>
  2. <script type="text/javascript">
  3. $(function () {
  4. $("#btnAjax").on("click",function () {
  5. $.ajax({
  6. dataType: 'json',
  7. url: 'return-void-ajax.do',
  8. data:{
  9. languageType:"ajax请求",
  10. workTime:20220303
  11. },
  12. // response 是形参 自定义的
  13. success: function (response) {
  14. alert("ajax请求响应结果:"+response.languageType+response.workTime);
  15. }
  16. });
  17. })
  18. })
  19. </script>

我们这里指定的响应数据格式是json ,这就是为什么一开始我们要添加相应依赖的原因了。这一步搞定,我们的控制器,就是服务端还没写呢,和之前一样,在MyController 中新增一个方法如下:

  1. /**
  2. * 控制器返回类型是 void 使用ajax请求演示结果
  3. *
  4. * @param response
  5. * @param languageType
  6. * @param workTime
  7. * @throws Exception
  8. */
  9. @RequestMapping(value = "/return-void-ajax.do")
  10. public void doAjax(HttpServletResponse response, String languageType, Integer workTime) throws Exception {
  11. Programmer programmer = new Programmer();
  12. programmer.setLanguageType(languageType);
  13. programmer.setWorkTime(workTime);
  14. ObjectMapper mapper = new ObjectMapper();
  15. String json = mapper.writeValueAsString(programmer);
  16. response.setContentType("application/json;charset=utf-8");
  17. PrintWriter pw = response.getWriter();
  18. pw.print(json);
  19. pw.flush();
  20. pw.close();
  21. }

在浏览器测试结果即可,能显示我们在ajax中指定的数据即是正确的。这块主要是熟悉流程,掌握程序执行顺序和逻辑。

执行流程分析

用户发起请求,首先到服务器(Tomcat),然后有中央调度器(DispatcherServlet)进行分发到不同的控制器(Controller),然后控制器执行对应的业务逻辑后返回数据视图给结果页。

控制器可以有多个,我们上面的例子只写了一个,且一个控制器中可以有多个方法处理不同的请求,同一个方法也可以处理多个请求。

总结

  • SpringMVC 其实可以理解为封装好的框架,简化我们的工作流程,提高开发效率
  • Spring体系很庞大,我们学习了前面的Spring Framework,现在学 SpringMVC 就感觉很轻松了,而且会感觉到后续的压力没那么重了

学编程,推荐首选 Java 语言,小编创建了一个专注Java的原创公众号推荐学java,大家可在微信搜索javaFirst 关注,一起开启 Java 旅途!