SpringMvc完整知识点二(完结)

SpringMVC获取请求参数

  • 环境准备工作等均省略,可详见快速入门,此处只写非共有部分代码

  • 该部分示例项目SpringMvcThree已上传至Gitee,可自行下载

  • 客户端请求参数的格式为:name=value&password=value... ...

  • 服务端想要获取请求参数有时需要对数据进行封装,SpringMVC可接收如下类型的参数

    • 普通数据类型(基本数据类型以及字符串类型)
    • POJO类型
    • 数组类型
    • 集合类型

获取普通数据类型

  • 注意

    • Controller中的业务方法的参数名称要与请求参数的名称一致,如图所示
      在这里插入图片描述

      在该截图中返回值为void,代表controller控制器中的该业务方法quickMethod9不回写数据,但仍然需要@ResponseBody注解,此时代表ResponseBody的响应体为空,也就是说前端页面为空。

    • 针对获取普通数据类型来说,参数未自动映射匹配有两种方法:

      • 在pom.xml文件中添加maven插件

      • 使用@RequestParam注解

      • 以上两种方式详见代码示例部分

代码示例

  • 此处只写controller控制器类,其余部分搭建可详见快速入门以及数据响应部分内容

  • 在controller包下创建UserController类,代码如下

    //将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping("/user")
    public class UserController {@ResponseBody@RequestMapping(value = "/quick1")public void save1(String username, int age) {System.out.println("name:" + name);System.out.println("age:" + age);}
    }
    

    此时运行截图如下,会报错

    在这里插入图片描述

    报错原因:错误说明 Spring MVC 无法推断参数名称,通常是因为编译时没有启用参数名的保留功能,或者方法参数缺少显式绑定的注解

  • 解决方法

    • 方案一: 启用参数名的保留功能,即在pom.xml文件中加上maven插件,插件代码如下:

      <plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><!-- maven插件版本 --><version>3.13.0</version> <configuration><!-- 所使用的Java版本 --><source>21</source> <target>21</target><compilerArgs><arg>-parameters</arg></compilerArgs></configuration>
      </plugin>
      

      在这里插入图片描述

    • 方案二: 为方法参数添加 @RequestParam 注解,明确指定参数名称,更改后的UserController类,代码如下

      • 注意:获取普通数据类型的参数以方案二为准,以此来避免影响后续获取其它类型参数的示例操作(即以pom.xml文件中未添加maven插件来演示后续获取各种类型的请求参数的操作,以此来证明其它类型均可自动映射匹配)
      //将Usercontroller放到Spring容器中
      @Controller
      @RequestMapping(value = "/user")
      public class UserController {@RequestMapping(value = "quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}
      }
      
    • 此时运行结果如下

      • GET请求:运行后前端页面为空白,但是控制台会有输出,如图所示

        在这里插入图片描述

      • :POST请求:运行后前端页面为空白,但是控制台会有输出,如图所示

        在这里插入图片描述

获取POJO类型

  • 注意

    • Controller中业务方法的参数为pojo类对象,且该pojo类中的的属性名要与请求参数的名称一致,如图所示

      在这里插入图片描述

    • 获取POJO类型时,不需要去添加maven插件或使用@RequestParam注解来使参数自动映射匹配,在本代码示例中的获取普通数据类型使用的是注解方式来进行参数匹配,以此来区别说明POJO类型不需要去添加maven插件或使用@RequestParam注解。

获取普通POJO类型代码示例

  • 创建pojo包,并在该包下创建一个User类,代码如下

    package at.guigu.pojo;public class User {private String name;private int age;public User() {}public User(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}@Overridepublic String toString() {return "User{" +"name='" + name + '\'' +", age=" + age +'}';}
    }
    
  • 在controller包下的UserController类中添加save2方法,完整代码如下:

    package at.guigu.controller;import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通pojo类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}
    }
    

在这里插入图片描述

获取嵌套POJO类型代码示例

  • 在pojo包中创建Address类、Brand类,并将其嵌套在Brand类中,此时Brand类及Address类简要代码如下

    public class Brand {private String brandName;private int brandPrice;private Address address;......
    }
    public class Address {private String province;private String city;......
    }
    
  • 在controller包下的UserController类中添加save22方法,完整代码如下:

    package at.guigu.controller;import at.guigu.pojo.Brand;
    import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通pojo类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}
    }
    

    在这里插入图片描述

获取JSON对象的POJO类型

  • 定义:前端传递的是JSON对象

  • 在controller包下的UserController类中添加save222方法,完整代码如下:

    传递JSON对象时参数需加上@RequestBody注解

    package at.guigu.controller;import at.guigu.pojo.Brand;
    import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通pojo类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}
    }
    

    在这里插入图片描述

获取数组类型

  • 注意:

    • Controller中业务方法的参数的数组名称要与请求参数的名称一致,如图所示

      在这里插入图片描述

    • 针对获取数组类型来说,参数未自动映射匹配有两种方法:

      • 在pom.xml文件中添加maven插件

      • 使用@RequestParam注解

      • 以上两种方法的操作与获取获取普通数据类型一致,可详见获取普通数据类型部分的代码示例,此处仅以注解形式示例

代码示例

  • 在controller包下的UserController类中添加save3方法,完整代码如下:

    package at.guigu.controller;import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Arrays;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通POJO类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}
    }
    

    在这里插入图片描述

获取集合类型—集合保存pojo类的对象

获取普通集合类型

  • 在controller包下的UserController类中添加savepre4方法,代码如下

    package at.guigu.controller;import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Arrays;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通POJO类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}// 获取普通集合类型@ResponseBody@RequestMapping("/quickpre4")public void savepre4(@RequestParam("strs") List<String> strs) throws IOException {System.out.println(strs);}
    }
    

    在这里插入图片描述

  • 针对获取数组类型来说,参数未自动映射匹配有两种方法:

    • 在pom.xml文件中添加maven插件

    • 使用@RequestParam注解

    • 以上两种方法的操作与获取获取普通数据类型一致,可详见获取普通数据类型部分的代码示例,此处仅以注解形式示例

获取JSON集合类型

  • 在controller包下的UserController类中添加savepre4方法,代码如下

    获取数组集合类型的参数需要用@RequestBody注解修饰

    package at.guigu.controller;import at.guigu.pojo.User;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Arrays;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通POJO类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}// 获取普通集合类型@ResponseBody@RequestMapping("/quickpre4")public void savepre4(@RequestParam("strs") List<String> strs) throws IOException {System.out.println(strs);}// 获取JSON集合类型@ResponseBody@RequestMapping("/quickpre44")public void savepre44(@RequestBody List<String> strs) throws IOException {System.out.println(strs);}
    }
    

    在这里插入图片描述

获取对象集合类型

  • 有三种获取方式
    • 使用POJO类进行集合封装,然后获取集合类型的请求参数
    • 不使用POJO类进行集合封装
方式一:使用POJO类进行集合封装
  • 注意
    • 获取集合参数时,要将集合参数封装到一个POJO类中才可以,也就是说,集合要封装到一个对象中(一般定义为VO类),作为这个对象里面的私有属性存在.此时就相当于获取POJO类型,详见代码示例
代码示例
  • Step1:POJO类中创建User类(代码详见获取POJO类型),创建VO类,代码如下

    package at.guigu.pojo;import java.util.List;public class VO {// 将想要获取的集合封装到对象中private List<User> userList;public List<User> getUserList() {return userList;}public void setUserList(List<User> userList) {this.userList = userList;}@Overridepublic String toString() {return "VO{" +"userList=" + userList +'}';}
    }
    
  • Step2:在controller包下的UserController类中添加save4方法,完整代码如下:

    package at.guigu.controller;
    import at.guigu.pojo.User;
    import at.guigu.pojo.VO;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Arrays;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通POJO类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}// 获取普通集合类型@ResponseBody@RequestMapping("/quickpre4")public void savepre4(@RequestParam("strs") List<String> strs) throws IOException {System.out.println(strs);}// 获取JSON集合类型@ResponseBody@RequestMapping("/quickpre44")public void savepre44(@RequestBody List<String> strs) throws IOException {System.out.println(strs);}// 获取对象集合类型方式一@ResponseBody@RequestMapping("/quick4")public void save4(VO vo) throws IOException {System.out.println(vo);}
    }
    

    此时相等于获取POJO类型:所以请求参数名称要与该POJO类(即VO类)中的属性名一致,而在VO类中该属性userList为一个集合,所以不仅要与userList一致,更要进一步与该集合里面存储的对象User的属性名样一致

  • Step3:以页面为例(post方式提交)web项目核心目录(即webapp)下创建一个form.jsp页面,代码如下:

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html><head><title>Title</title></head><body><%--将表单提交到控制器映射地址下的/user/quick4,也就是UserController类下的save4业务方法中--%><form action="/SpringMvcThree/user/quick4" method="post"><%--将第一个请求参数提交到userList集合中第一个元素的name属性中--%><input type="text" name="userList[0].name"><br/><%--将第一个请求参数提交到userList集合中第二个元素的age属性中--%><input type="text" name="userList[0].age"><br/><input type="text" name="userList[0].name"><br/><input type="text" name="userList[0].age"><br/><input type="submit" value="提交"></form></body>
    </html>
    

    注意:因为请求参数名称要与VO类中的集合属性名一致,又因为集合中存储的是POJO类对象(即User),而User类中有两个属性nameage,所以请求参数名称要与集合名.POJO类属性名一致,所以表单中name属性值为userList[0].name,它的含义就是集合中第一个元素的name属性。

    运行后截图如下所示:

    在这里插入图片描述

方式二:前端使用JSON数据发送集合
  • 注意
    • 当使用ajax提交表单时,可以指定contentType为json形式,然后在对应业务方法的参数位置使用@RequestBody注解就可以直接接收集合数据而不需要使用POJO进行封装
代码示例
  • Step1:在web项目核心目录(即webapp)下创建js目录,引入jquery源码文件jquery-3.7.1.js(官网自行下载)

  • Step2:web项目核心目录(即webapp)下创建一个ajax.jsp文件,并在该文件中引入jquery的源码文件,最终代码如下:

    <%--Created by IntelliJ IDEA.User: 10195Date: 2024/11/22Time: 16:04To change this template use File | Settings | File Templates.
    --%>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html><head><title>Title</title></head><body><script src="js/jquery-3.7.1.js"></script><script>// 创建核心对象var userList = new Array();userList.push({name:"zhangsna", age:15});userList.push({name:"lisi", age:16});$.ajax({type:"POST",url:"user/quick5",data:JSON.stringify(userList),contentType:"application/json;charset=utf-8"});</script></body>
    </html>
    
  • Step3:在springMVC的核心配置文件中添加如下代码:

    • 原因:在SpringMVC中默认会拦截对所有资源的请求,包括静态资源;若不单独配置则会使静态资源请求被误认为是需要交给核心前端控制器处理的业务请求,此时由于前端控制器无法找到与之对应的业务方法,从而导致资源无法正确加载,所以需要在SpringMVC的核心配置文件中对这些资源进行开放访问。

      <!--配置静态资源的路径映射,开放某些资源的访问-->
      <mvc:resources mapping="/js/**" location="/js/"/>
      
    • springMVC的核心配置文件完整代码如下:

      <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的前缀属性prefix的值设为/jsp/--><property name="prefix" value="/user/"></property><!--将InternalResourceViewResolver类中的前缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--mvc的注解驱动--><mvc:annotation-driven/><!--等同于配置处理器适配器--><!--<bean id="handlerAdapter" class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"><property name="messageConverters"><list><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/></list></property></bean>--><mvc:resources mapping="/js/**" location="/js/"/>
      </beans>
      
  • Step4:在controller包下的UserController类中添加save5方法,完整代码如下:

    package at.guigu.controller;import at.guigu.pojo.User;
    import at.guigu.pojo.VO;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestBody;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Arrays;
    import java.util.List;//将Usercontroller放到Spring容器中
    @Controller
    @RequestMapping(value = "/user")
    public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取普通POJO类型@ResponseBody@RequestMapping("/quick2")public void save2(User user) throws IOException {System.out.println(user);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}// 获取普通集合类型@ResponseBody@RequestMapping("/quickpre4")public void savepre4(@RequestParam("strs") List<String> strs) throws IOException {System.out.println(strs);}// 获取JSON集合类型@ResponseBody@RequestMapping("/quickpre44")public void savepre44(@RequestBody List<String> strs) throws IOException {System.out.println(strs);}// 获取对象集合类型方式一@ResponseBody@RequestMapping("/quick4")public void save4(VO vo) throws IOException {System.out.println(vo);}// 获取对象集合类型方式二@ResponseBody@RequestMapping("/quick5")public void save5(@RequestBody List<User> userList) throws IOException {System.out.println(userList);}
    }
    

    在这里插入图片描述

开放静态资源的请求访问(SpringMVC配置文件形式)

  • 注意

    • 在SpringMVC中默认会拦截对所有资源的请求,包括静态资源;若不单独配置则会使静态资源请求被误认为是需要交给核心前端控制器处理的业务请求,此时由于前端控制器无法找到与之对应的业务方法,从而导致资源无法正确加载,所以需要在SpringMVC的核心配置文件中对这些资源进行开放访问。
    • 开放静态资源的请求访问时,其路径映射不能与内部资源视图解析器一样,否则会报错
  • 通过<mvc:resources>标签进行资源路径配置标签常用属性如下:

    <mvc:resources>标签属性解释
    mapping告诉 Spring MVC,当用户请求某些特定 URL 时,这些请求是访问静态资源,而不是交给控制器处理。/js/**代表js后可以是多级url地址
    location定义资源在服务器上的实际存放位置
  • 由于每开放一种静态资源就要写一个该标签代码,所以可用如下代码代替

    <mvc:default-servlet-handler/>
    

    解释: 在SpringMVC中默认会拦截对所有资源的请求,包括静态资源;若不单独配置则会使静态资源请求被误认为是需要交给核心前端控制器处理的业务请求,此时由于前端控制器无法找到与之对应的业务方法,从而导致资源无法正确加载,此时就会交由Tomcat来找对应的静态资源,从而使得资源正确加载

    简要总结: SpringMVC框架无法找到对应资源时就会让原始容器Tomcat去找静态资源

  • 开放对图片、jquery文件等静态资源访问的代码示例如下

    <!--配置静态资源的路径映射,开放某些资源的访问-->
    <mvc:resources mapping="/js/**" location="/js/"/>
    <mvc:resources mapping="/img/**" location="/img/"/>
    

    等同于

    <mvc:default-servlet-handler/>
    

开放静态资源的请求访问方式一(SpringMVC类形式)

  • Step1:在config包下创建一个继承WebMvcConfigurationSupport的子类SpringMvcSupport,并为其加上@Configuration注解,代码如下

    package at.guigu.config;import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
    import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;// 静态资源的路径映射类
    @Configuration
    public class SpringMvcSupport extends WebMvcConfigurationSupport {@Overrideprotected void addResourceHandlers (ResourceHandlerRegistry registry) {// 当访问/pages/????时候,从/pages目录下查找内容// 配置静态资源的路径映射,开放某些资源的访问// 等同于<mvc:resources mapping="/js/**" location="/js/"/>registry.addResourceHandler("/xxx/**").addResourceLocations("/xxx/");// 等同于<mvc:resources mapping="/img/**" location="/img/"/>registry.addResourceHandler("/img/**").addResourceLocations("/img/");}
    }
    
  • Step2: 在SpringMVC核心配置类中用@ComponentScan注解扫描静态资源的路径映射类所在包,代码如下

    package at.guigu.config;import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.multipart.commons.CommonsMultipartResolver;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
    @Configuration
    // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
    // 加载controller对应的bean
    @ComponentScan({"at.guigu.controller", "at.guigu.config"})
    // 自动配置 Spring MVC 的各种特性,比如:类型转换器、mvc的注解驱动<mvc:annotation-driven/>、静态资源路径映射
    @EnableWebMvc
    public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/user/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}
    }
    

开放静态资源的请求访问方式二(SpringMVC类形式)

  • 让SpringMVC的核心配置类SpringMvcConfiguration实现WebMvcConfigurer接口,然后重写其中的addResourceHandlers方法即可,代码如下

    package at.guigu.config;import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.multipart.commons.CommonsMultipartResolver;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
    @Configuration
    // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
    // 加载controller对应的bean
    @ComponentScan("at.guigu.controller")
    // 自动配置 Spring MVC 的各种特性
    @EnableWebMvc
    public class SpringMvcConfiguration implements WebMvcConfigurer {// 依赖注入拦截器的bean@Autowiredprivate MyInterceptor myInterceptor;// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}// 配置静态资源路径@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {// 配置静态资源路径映射registry.addResourceHandler("/xxx/**").addResourceLocations("/xxx/");}
    }
    

GET请求参数乱码问题

  • Tomcat8.5版本之前会出现GET请求参数乱码问题,在其之后官方就已经给解决了

  • 若使用的是Tomcat7插件则会出现Get请求参数乱码问题,此时需要在pom.xml文件对应的Tomcat插件中配置UTF-8字符集,插件代码如下

    <plugins><!-- Tomcat插件 --><plugin><groupId>org.apache.tomcat.maven</groupId><artifactId>tomcat7-maven-plugin</artifactId><version>2.2</version><configuration><port>80</port><!--Tomcat端口号--><path>/</path><!--虚拟目录--><uriEncoding>UTF-8</uriEncoding><!--指定字符集--></configuration></plugin>
    </plugins>
    

POST请求参数乱码问题

SpringMVC配置文件形式

  • 当使用POST请求时,数据会出现乱码问题(可详见使用POJO类进行集合封装的代码示例),解决方法:

    • 在web项目核心目录(即webapp)下的WEB-INF中的web.xml中配置一个全局过滤器来进行编码的过滤,代码如下

      注意:<filter>以及<filter-mapping>标签需要写到<linstener>标签前,因为web.xml中有严格的标签顺序

      <!--全局过滤器-->
      <filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><init-param><param-name>encoding</param-name><param-value>UTF-8</param-value></init-param>
      </filter>
      <filter-mapping><filter-name>CharacterEncodingFilter</filter-name><url-pattern>/*</url-pattern>
      </filter-mapping>
      
    • 此时再次运行使用POJO类进行集合封装的代码示例后就不会出现乱码问题了,运行截图如下

      在这里插入图片描述

SpringMVC配置类形式

  • 当使用POST请求时,数据会出现乱码问题(可详见使用POJO类进行集合封装的代码示例),解决方法:

    • 在web.xml文件对应的配置类中指定字符过滤器,代码如下

      package at.guigu.config;import org.springframework.web.context.WebApplicationContext;
      import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
      import org.springframework.web.filter.CharacterEncodingFilter;
      import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
      import org.springframework.web.servlet.support.AbstractDispatcherServletInitializer;import javax.servlet.Filter;// 定义一个Servlet容器启动的配置类,来加载Spring的配置
      public class ServletContainerInitConfiguration extends AbstractAnnotationConfigDispatcherServletInitializer {// 加载SpringMVC的配置@Overrideprotected Class<?>[] getServletConfigClasses() {return new Class[]{SpringMvcConfiguration.class};}// 设置哪些请求归SpringMVC处理@Overrideprotected String[] getServletMappings() {return new String[]{"/"};}// 加载非SringMVC的配置,比如:加载Spring的配置@Overrideprotected Class<?>[] getRootConfigClasses() {return new Class[]{SpringConfiguration.class};}// POST请求乱码处理:指定字符过滤器@Overrideprotected Filter[] getServletFilters() {CharacterEncodingFilter filter = new CharacterEncodingFilter();filter.setEncoding("UTF-8");return new Filter[]{filter};}
      }/* 等同于
      // 定义一个Servlet容器启动的配置类,来加载Spring的配置
      public class ServletContainerInitConfiguration extends AbstractDispatcherServletInitializer {// 加载SpringMVC的配置@Overrideprotected WebApplicationContext createServletApplicationContext() {AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();context.register(SpringMvcConfiguration.class);return context;}// 设置哪些请求归SpringMVC处理@Overrideprotected String[] getServletMappings() {// 代表将所有请求都交给前端控制器处理return new String[]{"/"};}// 加载非SringMVC的配置,比如:加载Spring的配置@Overrideprotected WebApplicationContext createRootApplicationContext() {AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();context.register(SpringConfiguration.class);return context;}// POST请求乱码处理:指定字符过滤器@Overrideprotected Filter[] getServletFilters() {CharacterEncodingFilter filter = new CharacterEncodingFilter();filter.setEncoding("UTF-8");return new Filter[]{filter};}
      }*/

@requestParam注解

@requestParam注解属性解释
value请求参数的名称
required该注解指定的请求参数是否必须存在,默认为true,提交时若该参数不存在则会报错
defaultValue当该注解没有指定请求参数时,则使用指定默认值
  • 定义:当请求参数的名称与Controller控制器中对应的业务方法的参数名称不一致时,将它们显式的绑定到一块

    在这里插入图片描述

  • 代码示例如下:

    • 在controller包下的UserController类中添加save6方法,完整代码如下:

      package at.guigu.controller;
      import at.guigu.pojo.User;
      import at.guigu.pojo.VO;
      import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestBody;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.RequestParam;
      import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
      import java.util.Arrays;
      import java.util.List;//将Usercontroller放到Spring容器中
      @Controller
      @RequestMapping(value = "/user")
      public class UserController {// 获取普通数据类型@RequestMapping(value = "/quick1")@ResponseBodypublic void save1(@RequestParam("username")String username, @RequestParam("age")int age) throws IOException {System.out.println(username);System.out.println(age);}// 获取嵌套POJO类型@ResponseBody@RequestMapping("/quick22")public void save22(Brand brand) throws IOException {System.out.println(brand);}// 获取JSON对象的POJO类型@ResponseBody@RequestMapping("/quick222")public void save222(@RequestBody User user) throws IOException {System.out.println(user);}// 获取数组类型@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("strs")String[] strs) throws IOException {// 由于直接打印数组时只会打印出其地址,所以将其转为List集合输出到控制台System.out.println(Arrays.asList(strs));}// 获取普通集合类型@ResponseBody@RequestMapping("/quickpre4")public void savepre4(@RequestParam("strs") List<String> strs) throws IOException {System.out.println(strs);}// 获取JSON集合类型@ResponseBody@RequestMapping("/quickpre44")public void savepre44(@RequestBody List<String> strs) throws IOException {System.out.println(strs);}// 获取对象集合类型方式一@ResponseBody@RequestMapping("/quick4")public void save4(VO vo) throws IOException {System.out.println(vo);}// 获取对象集合类型方式二@ResponseBody@RequestMapping("/quick5")public void save5(@RequestBody List<User> userList) throws IOException {System.out.println(userList);}// 测试@RequestParam注解@RequestMapping(value = "/quick1")@ResponseBodypublic void save6(@RequestParam(value = "name", required = false, defaultValue = "zhangzhang")String username) throws IOException {System.out.println(username);}
      }
      
      • 请求参数名与业务方法的参数名不一致

        在这里插入图片描述

      • 不写请求参数,则会将指定默认值赋值给业务方法对应的参数

        在这里插入图片描述

自定义类型转换器

  • SpringMVC默认已经提供了一些常用的类型转换器,比如将客户端提交的字符串转换成int类型进行参数设置。但不是所有是数据类型都提供了转换器,比如日期类型的数据,此时就需要自定义转换器

  • 注意

    • 参数若不会自动映射匹配,所以可使用如下两种方法
      • 使用@RequestParam注解
      • 在pom.xml文件中添加maven插件
      • 以上两种方式代码示例详见获取普通数据类型

日期类型参数传递示例一(SpringMVC配置文件形式)

本代码示例以日期类型为例:将日期封装为yyyy-MM-dd形式

注意:日期默认封装格式为yyyy/MM/dd形式,若请求参数中为yyyy-MM-dd形式则会报错,所以本示例是将其转换为常用的yyyy-MM-dd形式

  • 开发步骤

    • 自定义一个实现Converter接口的转换器类

    • 在SpringMVC的核心配置文件中声明转换器类

    • 在核心配置文件中用<annotation-driven>(即mvc的注解驱动)标签中的conversion-service属性引用转换器类

  • Step1: 创建一个与三层架构包同级的converter包,然后定义一个实现Converter接口的类DataConverter,并定义该类的泛型为<String, Date>,然后重写其中的convert方法,在该方法中将日期转换成指定格式的日期对象并返回

    • 注意:
      • 该接口是org.springframework.core.convert.converter.Converter包下的
      • <String, Date>中第一个参数String为请求参数字符串;第二个参数Date为要转换到的类型
      • convert方法的参数为客户端的请求参数
    package at.guigu.converter;import org.springframework.core.convert.converter.Converter;import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.Date;public class DataConverter implements Converter<String, Date> {@Overridepublic Date convert(String source) {// 将日期字符串转换成指定的日期对象并返回SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");Date date = null;try {date = format.parse(source);} catch (ParseException e) {e.printStackTrace();}return date;}
    }
    
  • Step2: 在SpringMVC的核心配置文件中声明转换器类,核心配置文件代码如下

    <!--声明转换器类-->
    <bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean"><property name="converters"><list><bean class="at.guigu.converter.DataConverter"></bean></list></property>
    </bean>
    
  • Step3: 在SpringMVC的核心配置文件中用<annotation-driven>标签中的conversion-service属性引用转换器类的id,代码如下

    <mvc:annotation-driven conversion-service="conversionService"/>
    

    SpringMVC的核心配置文件代码如下

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的前缀属性prefix的值设为/jsp/--><property name="prefix" value="/user/"></property><!--将InternalResourceViewResolver类中的前缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--mvc的注解驱动--><!--利用conversion-service来引用转换器类,属性值为转换器对应的id--><mvc:annotation-driven conversion-service="conversionService"/><!--等同于配置处理器适配器--><!--<bean id="handlerAdapter" class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"><property name="messageConverters"><list><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/></list></property></bean>--><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--声明转换器类--><bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean"><property name="converters"><list><bean class="at.guigu.converter.DataConverter"></bean></list></property></bean>
    </beans>
    
  • Step4: 在controller包下创建UserControllerThree类并添加save1方法,完整代码如下:

    package at.guigu.controller;import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Date;@Controller
    @RequestMapping("/user3")
    public class UserControllerThree {@ResponseBody@RequestMapping("/quickk1")public void save1(@RequestParam("date") Date date) throws IOException {System.out.println(date);}
    }
    

    运行截图如下

    在这里插入图片描述

日期类型参数传递示例二(SpringMVC配置类形式)

  • 日期类型数据基于系统不同格式也不尽相同,所以我们在接收形参时,可以根据不同的日期格式设置不同的接收方式
    2088-08-18
    2088/08/18
    08/18/2088

  • 注意:日期类型参数传递除了使用自定义类型转换器外(详见自定义类型转换器中的内容),还可以使用注解形式,步骤如下

    • 在控制器给对应方法的形参加上@DateTimeFormat(pattern)注解

      • 它内部依赖的是Converter接口
    • 在SpringMVC的核心配置类上加上@EnableWebMvc注解,它可以根据类型匹配对应的类型转换器,代替了SpringMVC配置文件形式中的一下两步:

      • 在SpringMVC的核心配置文件中声明转换器类

      • 在核心配置文件中用<annotation-driven>(即mvc的注解驱动)标签中的conversion-service属性引用转换器类

  • Step1:MvcReqClaDemo项目示例中的controller包下创建UserControllerThree类并添加save1方法,完整代码如下

    package at.guigu.controller;import org.springframework.format.annotation.DateTimeFormat;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;import java.io.IOException;
    import java.util.Date;// 自定义类型转换器
    @Controller
    @RequestMapping("/user3")
    public class UserControllerThree {@ResponseBody@RequestMapping("/quickk1")public void save1(@RequestParam("date") Date date,@DateTimeFormat(pattern = "yyyy-MM-dd") @RequestParam("date1") Date date1,@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") @RequestParam("date2") Date date2) throws IOException {System.out.println(date);System.out.println(date1);System.out.println(date2);}
    }
    
  • Step2: SpringMVC核心类添加上@EnableWebMvc注解,代码如下

    package at.guigu.config;import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
    @Configuration
    // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
    // 加载controller对应的bean
    @ComponentScan("at.guigu.controller")
    // 自动配置 Spring MVC 的各种特性,比如:类型转换器、mvc的注解驱动<mvc:annotation-driven/>
    @EnableWebMvc
    public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/user/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}
    }
    

    在这里插入图片描述

获取Servlet的相关API

  • SpringMVC支持使用原始ServletAPI对象作为控制器方法的参数进行注入,常用对象有

    • HttpServletRequest
    • HttpServletResponse
    • HttpSession
  • 获取方式: 只需要将想要的Servlet的API作为控制器对应方法的参数即可

    • 方法一般是谁调用谁传参,因为业务方法是SpringMVC框架调用的,所以SpringMVC会自动根据方法的参数进行注入
  • 测试示例

    • 在controller包下创建UserControllerFour类并添加save1方法,完整代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.ResponseBody;import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      import javax.servlet.http.HttpSession;@Controller
      @RequestMapping("/user4")
      public class UserControllerFour {@ResponseBody@RequestMapping("/quick1")public void save1(HttpServletRequest req, HttpServletResponse res, HttpSession hs) throws Exception{System.out.println(req);System.out.println(res);System.out.println(hs);}
      }
      

      在这里插入图片描述

获取请求头信息

  • 利用@RequestHeader注解来获取请求头信息,相当于web阶段所学的request.getHeader(name)方法,可详见会话跟踪技术部分内容

    @RequestHeader注解属性解释
    value请求头名称
    required是否必须携带该请求头,默认为true,即必须携带该请求头才能访问这个资源
  • 测试示例

    • 在controller包下创建UserControllerFive类并添加save1方法,代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.ResponseBody;@Controller
      @RequestMapping("/user5")
      public class UserControllerFive {@ResponseBody@RequestMapping("/quick1")public void save1() throws Exception{}
      }
      

      运行后通过开发者工具可看到请求头信息,如图所示

      在这里插入图片描述

    • 假设现在获取请求头user-agent的信息,则代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestHeader;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.ResponseBody;@Controller
      @RequestMapping("/user5")
      public class UserControllerFive {@ResponseBody@RequestMapping("/quick1")public void save1(@RequestHeader(value = "User-Agent", required = false) String user_agent) throws Exception{System.out.println(user_agent);}
      }
      

      在这里插入图片描述

获取指定Cookie请求头信息

  • @RequestHeader只能根据如图所示1中的请求头来获取2中的信息,而有些请求头后的信息有键值对,比如Cookie中又有很多键值对Cookie,此时若想获取Cookie里面的小Cookie的话@RequestHeader注解就会失效。

    在这里插入图片描述

  • Cookie不只有一个,所以属于特殊请求头,如图所示
    在这里插入图片描述

  • @CookieValue获取指定Cookie的值

    @CookieValue注解属性解释
    valueCookie名称
    required是否必须携带该Cookie,默认为true,即必须携带该Cookie才能访问这个资源
  • 代码示例(此处以JSESSIONID这个Cookie为例)

    • UserControllerFive类中添加save2方法,代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.CookieValue;
      import org.springframework.web.bind.annotation.RequestHeader;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.ResponseBody;@Controller
      @RequestMapping("/user5")
      public class UserControllerFive {// 获取普通请求头@ResponseBody@RequestMapping("/quick1")public void save1(@RequestHeader(value = "User-Agent", required = false) String user_agent) throws Exception{System.out.println(user_agent);}// 获取特殊请求头:获取指定Cookie@ResponseBody@RequestMapping("/quick2")public void save2(@CookieValue(value = "JSESSIONID", required = false) String jessionid) throws Exception{System.out.println(jessionid);}
      }
      

      在这里插入图片描述

  • 注意:除以上注解之外,也可使用通用方式(即使用HttpServletRequest接口中的方法)来获取请求头信息,可详见会话跟踪技术部分内容

文件上传(获取文件)

  • 文件上传客户端三要素

    • 表单项type="file"
    • 表单提交方式为post
    • 表单的enctype属性是多部分表单形式,即enctype="mulipart/form-data"

    在这里插入图片描述

  • 注意

    • 当form表单修改为多部分表单时,request.getParameter(String name)会失效,因为该方法只能获取单个参数值

      • 该方法是根据键名来获取参数值,具体解释可详见可见Web&Http&Servlet&Request&Response部分内容
    • 默认情况下,enctype="application/x-www-form-urlencoded",此时form表单的正文内容是key=value&key=value&key=value

    • enctype="mulipart/form-data"时,请求正文内容就会变成多部分形式,此时能够获取表单的所有数据,如图所示

      在这里插入图片描述

  • 文件上传步骤

    • 在pom.xml文件中导入坐标:fileupload和io两个坐标

      <!--fileupload坐标-->
      <dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.5</version>
      </dependency><!--io坐标-->
      <dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.17.0</version>
      </dependency>
      
    • 在SpringMVC的核心配置文件中配置文件上传解析器

    • 编写文件上传代码

  • 单文件上传和多文件上传的公共步骤(后续代码演示不在演示公共步骤)

    • 在pom.xml文件中导入fileupload和io两个坐标,文件完整代码如下:

      <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"><modelVersion>4.0.0</modelVersion><parent><groupId>org.example</groupId><artifactId>SpringMvcDemo</artifactId><version>1.0-SNAPSHOT</version></parent><artifactId>SpringMvcThree</artifactId><packaging>war</packaging><name>SpringMvcThree Maven Webapp</name><url>http://maven.apache.org</url><dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>3.8.1</version><scope>test</scope></dependency><!--===================Spring相关坐标=======================--><!--spring坐标--><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>6.1.6</version></dependency><!--spring-web --><dependency><groupId>org.springframework</groupId><artifactId>spring-web</artifactId><version>5.2.25.RELEASE</version></dependency><!--spring-test坐标--><dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>6.1.6</version><scope>test</scope></dependency><!--Annotation坐标--><dependency><groupId>javax.annotation</groupId><artifactId>javax.annotation-api</artifactId><version>1.3.2</version></dependency><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.13.2</version><scope>test</scope></dependency><!-- servlet--><dependency><groupId>javax.servlet</groupId><artifactId>javax.servlet-api</artifactId><version>4.0.1</version><scope>provided</scope></dependency><!--jsp--><dependency><groupId>javax.servlet.jsp</groupId><artifactId>javax.servlet.jsp-api</artifactId><version>2.3.3</version><scope>provided</scope></dependency><!--===================SpringMVC相关坐标=======================--><!--spring-webmvc--><dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.2.25.RELEASE</version></dependency><!--jackson-core--><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-core</artifactId><version>2.17.1</version></dependency><!--jackson-databind--><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.17.1</version></dependency><!--jackson-annotations--><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-annotations</artifactId><version>2.17.1</version></dependency><!--=====================数据库相关坐标=========================--><!--mysql坐标--><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>8.0.33</version></dependency><!--druid坐标--><dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.2.18</version></dependency><!--c3p0坐标--><dependency><groupId>com.mchange</groupId><artifactId>c3p0</artifactId><version>0.9.5.5</version></dependency><!--=====================MyBatis相关坐标=========================--><!--spring-jdbc--><dependency><groupId>org.springframework</groupId><artifactId>spring-jdbc</artifactId><version>6.1.10</version></dependency><!--mybatis-spring--><dependency><groupId>org.mybatis</groupId><artifactId>mybatis-spring</artifactId><version>3.0.3</version></dependency><!--MyBatis坐标--><dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.5.16</version></dependency><!--=====================文件上传相关坐标=========================--><!--fileupload坐标--><dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.5</version></dependency><!--io坐标--><dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.17.0</version></dependency></dependencies><build><finalName>SpringMvcThree</finalName><plugins><!-- Tomcat插件 --><plugin><groupId>org.apache.tomcat.maven</groupId><artifactId>tomcat7-maven-plugin</artifactId><version>2.2</version></plugin><!--<plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId>&lt;!&ndash; maven插件版本 &ndash;&gt;<version>3.13.0</version><configuration>&lt;!&ndash; Java版本 &ndash;&gt;<source>21</source><compilerArgs><arg>-parameters</arg></compilerArgs></configuration></plugin>--></plugins></build>
      </project>
      
    • 在SpringMVC的核心配置文件中配置文件上传解析器,代码如下

      SpringMVC核心配置文件代码如下

      <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的前缀属性prefix的值设为/jsp/--><property name="prefix" value="/user/"></property><!--将InternalResourceViewResolver类中的前缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--mvc的注解驱动--><mvc:annotation-driven conversion-service="conversionService"/><!--等同于配置处理器适配器--><!--<bean id="handlerAdapter" class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"><property name="messageConverters"><list><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/></list></property></bean>--><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--声明转换器类--><bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean"><property name="converters"><list><bean class="at.guigu.converter.DataConverter"></bean></list></property></bean><!--配置文件上传解析器--><bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><!--所上传文件的编码类型--><property name="defaultEncoding" value="UTF-8"/><!--所上传的单个文件的大小--><property name="maxUploadSizePerFile" value="500000"/><!--所上传的总文件的大小--><property name="maxUploadSize" value="5000000"/></bean>
      </beans>
      

      SpringMVC核心配置文件对应的核心配置类代码如下

      package at.guigu.config;import org.springframework.context.annotation.Bean;
      import org.springframework.context.annotation.ComponentScan;
      import org.springframework.context.annotation.Configuration;
      import org.springframework.context.annotation.Import;
      import org.springframework.web.multipart.commons.CommonsMultipartResolver;
      import org.springframework.web.servlet.config.annotation.EnableWebMvc;
      import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
      @Configuration
      // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
      // 加载controller对应的bean
      @ComponentScan("at.guigu.controller")
      // 自动配置 Spring MVC 的各种特性,比如:类型转换器、mvc的注解驱动<mvc:annotation-driven/>、静态资源路径映射
      @EnableWebMvc
      // 引入配置静态资源的路径映射类
      @Import(SpringMvcSupport.class)
      public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/user/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}// 配置文件上传解析器@Beanpublic CommonsMultipartResolver multipartResolver() {CommonsMultipartResolver resolver = new CommonsMultipartResolver();resolver.setDefaultEncoding("UTF-8"); // 所上传文件的编码类型resolver.setMaxUploadSizePerFile(500000);// 所上传的单个文件的大小resolver.setMaxUploadSize(5000000);// 所上传的总文件的大小return resolver;}
      }
      

      在这里插入图片描述

单文件上传示例

  • 在web项目核心目录(即webapp)下创建文件upload.jsp,文件代码如下:

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html><head><title>Title</title></head><body><form action="/SpringMvcThree/user6/quick1" method="post" enctype="multipart/form-data">名称<input type="text" name = "username"><br/>文件<input type="file" name = "uploadFile"><br/><input type="submit" value="提交"><br/></form></body>
    </html>
    
  • 在controller包下创建UserControllerSix类并添加save1方法,完整代码如下:

    package at.guigu.controller;import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.ResponseBody;
    import org.springframework.web.multipart.MultipartFile;import java.io.File;
    import java.io.IOException;@Controller
    @RequestMapping("/user6")
    public class UserControllerSix {@ResponseBody@RequestMapping("/quick1")public void save1(@RequestParam("username") String username, @RequestParam("uploadFile") MultipartFile uploadFile) throws IOException {System.out.println(username);System.out.println(uploadFile);// 获取上传文件名String originalFilename = uploadFile.getOriginalFilename();// 保存文件uploadFile.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename));}
    }
    

    在这里插入图片描述

    • 注意

      • 业务方法的参数名要与请求参数名一致,由于表单上传的文件会被SpringMVC封装成一个MultipartFile对象,且对象名为表单中所定义的文件的name属性值,所以对应业务方法中第二个参数的名为uploadFile

      • 单文件上传时,参数不会自动映射匹配,解决方法有两种:

        • 在pom.xml文件中添加maven插件

        • 使用@RequestParam注解

        • 以上两种方式详见获取普通数据类型的代码示例部分

多文件上传示例

  • 方式一:文件name属性的属性名不一样

    • 在web项目核心目录(即webapp)下创建文件upload2.jsp,文件代码如下:

      <%@ page contentType="text/html;charset=UTF-8" language="java" %>
      <html><head><title>Title</title></head><body><form action="/SpringMvcThree/user6/quick2" method="post" enctype="multipart/form-data">名称<input type="text" name = "username"><br/>文件1<input type="file" name = "uploadFile1"><br/>文件2<input type="file" name = "uploadFile2"><br/><input type="submit" value="提交"><br/></form></body>
      </html>
      
    • UserControllerSix类中添加save2方法,完整代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.RequestParam;
      import org.springframework.web.bind.annotation.ResponseBody;
      import org.springframework.web.multipart.MultipartFile;import java.io.File;
      import java.io.IOException;@Controller
      @RequestMapping("/user6")
      public class UserControllerSix {// 单文件上传@ResponseBody@RequestMapping("/quick1")public void save1(@RequestParam("username") String username, @RequestParam("uploadFile") MultipartFile uploadFile) throws IOException {System.out.println(username);System.out.println(uploadFile);// 获取上传文件名String originalFilename = uploadFile.getOriginalFilename();// 保存文件uploadFile.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename));}//多文件上传 方式一@ResponseBody@RequestMapping("/quick2")public void save2(@RequestParam("username") String username, @RequestParam("uploadFile1") MultipartFile uploadFile1, @RequestParam("uploadFile2") MultipartFile uploadFile2) throws IOException {System.out.println(username);System.out.println(uploadFile1);System.out.println(uploadFile2);// 获取上传文件1的文件名String originalFilename1 = uploadFile1.getOriginalFilename();// 保存文件1uploadFile1.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename1));// 获取上传文件2的文件名String originalFilename2 = uploadFile2.getOriginalFilename();// 保存文件2uploadFile2.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename2));}
      }
      

      在这里插入图片描述

  • 方式二:文件name属性的属性名一样,此时用MultipartFile对象数组

    • 在web项目核心目录(即webapp)下创建文件upload3.jsp,文件代码如下:

      <%@ page contentType="text/html;charset=UTF-8" language="java" %>
      <html><head><title>Title</title></head><body><form action="/SpringMvcThree/user6/quick3" method="post" enctype="multipart/form-data">名称<input type="text" name = "username"><br/>文件1<input type="file" name = "uploadFiles"><br/>文件2<input type="file" name = "uploadFiles"><br/><input type="submit" value="提交"><br/></form></body>
      </html>
      
    • UserControllerSix类中添加save2方法,完整代码如下:

      package at.guigu.controller;import org.springframework.stereotype.Controller;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.RequestParam;
      import org.springframework.web.bind.annotation.ResponseBody;
      import org.springframework.web.multipart.MultipartFile;import java.io.File;
      import java.io.IOException;@Controller
      @RequestMapping("/user6")
      public class UserControllerSix {// 单文件上传@ResponseBody@RequestMapping("/quick1")public void save1(@RequestParam("username") String username, @RequestParam("uploadFile") MultipartFile uploadFile) throws IOException {System.out.println(username);System.out.println(uploadFile);// 获取上传文件名String originalFilename = uploadFile.getOriginalFilename();// 保存文件uploadFile.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename));}//多文件上传 方式一@ResponseBody@RequestMapping("/quick2")public void save2(@RequestParam("username") String username, @RequestParam("uploadFile1") MultipartFile uploadFile1, @RequestParam("uploadFile2") MultipartFile uploadFile2) throws IOException {System.out.println(username);System.out.println(uploadFile1);System.out.println(uploadFile2);// 获取上传文件1的文件名String originalFilename1 = uploadFile1.getOriginalFilename();// 保存文件1uploadFile1.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename1));// 获取上传文件2的文件名String originalFilename2 = uploadFile2.getOriginalFilename();// 保存文件2uploadFile2.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename2));}// 多文件上传 方式二@ResponseBody@RequestMapping("/quick3")public void save3(@RequestParam("username") String username, @RequestParam("uploadFiles") MultipartFile[] uploadFiles) throws IOException {System.out.println(username);for (MultipartFile file : uploadFiles) {System.out.println(file);// 获取上传文件名String originalFilename = file.getOriginalFilename();// 保存文件file.transferTo(new File("F:\\node\\idea\\test\\" + originalFilename));}}
      }
      

      在这里插入图片描述

获取Rest风格的请求参数

  • Restful定义

    • 根据REST风格对资源进行访问称为Restful
  • Rest风格

    • Rest是一种软件架构风格、设计风格,而不是标准

    • 它只是提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件

    • 基于该风格的软件会更简洁更有层次,更易实现缓存机制等

    • 可以利用Rest风格来省略请求参数名的书写(即简化书写),比如

      http://localhost:8080/SpringMvcThree/user/quick6?name=zhangsan
      

      可改写为

      http://localhost:8080/SpringMvcThree/user/quick6/zhangsan
      
    • 隐藏资源的访问行为,无法通过地址得知对资源进行的是什么操作,比如

      http://localhost/user/saveUser
      http://localhost/user/deleteUser
      http://localhost/user/updateUser
      

      均可改写为

      http://localhost/user
      

      此时会根据请求方式来判断执行的是哪个方法

  • 当使用Rest风格来获取请求参数时,需要在方法的@RequestMapping注解中用占位符指明映射地址后的为请求参数;同时用@PathVariable注解来修饰业务方法中的参数,且该注解的value值要与@RequestMapping注解中的占位符名称一致

    在这里插入图片描述

  • Rest风格请求使用的是url+请求方式来表示一次请求目的,Http协议中有4种操作方式:

    • GET:用于获取资源
      • /users/1 GET:获取id=1的user
    • POST:用于新增/保存资源
      • /users POST:新增/保存user
    • PUT:用于修改/更新资源
      • /users/1 PUT:修改/更新id=1的user
    • DELETE:用于删除资源
      • /users/1 DELETE:删除id=1的user
    • 注意: 描述模块的名称通常使用复数,也就是加s的格式描述,表示此类资源,而非单个资源,例如:users、books、accounts……
  • 用到的主要注解

    • @RequestMapping(value, method)method用来指定操作方式
    • @PathVatiable
    • @GetMapping(value):设置当前控制器方法Get请求访问路径以及Restful风格的动作,等同于@RequestMapping(value, method=RequestMethod.Get)
    • @PostMapping(value):设置当前控制器方法Post请求访问路径以及Restful风格的动作,等同于@RequestMapping(value, method=RequestMethod.Post)
    • @PutMapping(value):设置当前控制器方法Put请求访问路径以及Restful风格的动作,等同于@RequestMapping(value, method=RequestMethod.Put)
    • @DeleteMapping(value):设置当前控制器方法Delete请求访问路径以及Restful风格的动作,等同于@RequestMapping(value, method=RequestMethod.Delete)
    • 以上注解可详见SpringMVC注解解析部分内容

简单代码示例

  • 在controller包下创建UserControllerTwo类,并在该类中添加save1方法,完整代码如下:

    package at.guigu.controller;import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.ResponseBody;@Controller
    @RequestMapping("/user2")
    public class UserControllerTwo {// Restful风格的请求参数@ResponseBody@RequestMapping("/quickk1/{name}")public void save1(@PathVariable(value = "name") String userName) {System.out.println(userName);}
    }
    

    在这里插入图片描述

SpringMVC拦截器(interceptor)

  • 定义

    • SpringMVC拦截器(interceptor)是一种动态拦截方法调用的机制,在SpringMVC中动态拦截控制器方法的执行
    • SpringMVC拦截器(interceptor)类似于Servlet开发中所使用的过滤器(Filter),用于对处理器进行预处理和后处理
    • 拦截器是AOP思想的具体体现
  • 作用

    • 在指定方法调用的前后执行预先设定的代码
    • 阻止原始方法的执行
  • 在实际应用中,会将拦截器(interceptor)按一定顺序联结成一条链,该链被称为拦截器链(Interceptor Chain)

    • 在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用
  • 拦截器(Interceptor)和过滤器(Filter)的区别

    区别过滤器拦截器
    适用范围是servlet规范中的一部分,任何JavaWeb工程都可以使用是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用
    拦截范围在url-pattern中配置了/*后,会对所有要访问的资源进行拦截增强<mvc:mapping path=""/>中配置了/**之后,也会对所有资源进行拦截增强,但是可以通过<mvc:exclude-mapping path=""/>标签排除不需要拦截的资源
  • 步骤

    • 创建实现HandlerInterceptor接口的实现类,并重写它的三个方法
    • 在SpringMVC的核心配置文件中配置拦截器
    • 测试拦截器的拦截效果

快速入门

配置文件形式

  • 注意:此处仅弄一个简单环境来测试拦截器,所以准备工作均省略,可详见快速入门以及SpringMVC数据响应部分内容,完整框架如图所示

    在这里插入图片描述

  • 以上形式还未配置拦截器,所以运行后前面能接收到响应如图所示,现要求进行拦截器配置

    在这里插入图片描述

  • Step1: 创建一个与三层架构包同级的interceptor包并在该包下创建一个实现HandlerInterceptor接口的拦截器类,代码如下

    package at.guigu.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");return false;}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也已返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
    }
    
  • Step2: 在SpringMVC的核心配置文件中配置拦截器

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--mvc的注解驱动--><mvc:annotation-driven/><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的视图名称前缀属性prefix的值设为/jsp/--><property name="prefix" value="/jsp/"></property><!--将InternalResourceViewResolver类中的视图名称后缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--配置拦截器--><mvc:interceptors><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptor"/></mvc:interceptor></mvc:interceptors>
    </beans>
    

    配置文件形式运行后截图如下

    在这里插入图片描述

  • Step3: 测试拦截器拦截效果

    • 在以上运行截图中,只执行了preHandle方法是因为当它的返回值为true表示请求可以交给Controller来执行原始方法;反之则请求被拦截,此时只会执行preHandle方法

    • 此时若将其返回值改true,代码如下

      package at.guigu.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
      import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");return true;}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也以返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
      }
      

      此时的运行截图如下

      在这里插入图片描述

配置类形式方法一

  • 注意:此处仅弄一个简单环境来测试拦截器,所以准备工作均省略,可详见快速入门以及SpringMVC数据响应部分内容,完整框架如图所示

    在这里插入图片描述

  • 以上形式还未配置拦截器,所以运行后前面能接收到响应如图所示,现要求进行拦截器配置

    在这里插入图片描述

  • Step1: 创建一个与三层架构包同级的interceptor包并在该包下创建一个实现HandlerInterceptor接口的拦截器类,代码如下

    注意:要给该类加上@Component注解(作用:实例化bean)

    package at.guigu.interceptor;import org.springframework.stereotype.Component;
    import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;@Component
    public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");return false;}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也以返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
    }
    
  • Step2: 在config包下创建配置静态资源以及拦截器的路径映射类,即继承WebMvcConfigurationSupport类的子类SpringMvcSupport,代码如下:

    • 依赖注入MyInterceptor的bean
    • 重写addInterceptors方法
    package at.guigu.config;import at.guigu.interceptor.MyInterceptor;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
    import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
    import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;// 配置静态资源以及拦截器的路径映射
    @Configuration
    public class SpringMvcSupport extends WebMvcConfigurationSupport {@Autowiredprivate MyInterceptor myInterceptor;@Overrideprotected void addResourceHandlers (ResourceHandlerRegistry registry) {// 配置静态资源路径映射registry.addResourceHandler("/xxx/**").addResourceLocations("/xxx/");}@Overrideprotected void addInterceptors(InterceptorRegistry registry) {//addInterceptor定义拦截器的具体实现类//addPathPatterns的参数代表拦截所有请求路径registry.addInterceptor(myInterceptor).addPathPatterns("/**");}
    }
    
  • Step3: 在SpringMVC的核心配置类中利用@ComponentScan注解扫描拦截器类MyInterceptor以及SpringMvcSupport类所在包

    package at.guigu.config;import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
    @Configuration
    // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
    // 加载controller对应的bean
    @ComponentScan({"at.guigu.controller", "at.guigu.interceptor", "at.guigu.config"})
    // 自动配置 Spring MVC 的各种特性
    @EnableWebMvc
    public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}
    }
    

    配置类形式运行后截图如下

    在这里插入图片描述

    Step4: 测试拦截器拦截效果

    • 在以上运行截图中,只执行了preHandle方法是因为当它的返回值为true表示请求可以交给Controller来执行原始方法;反之则请求被拦截,此时只会执行preHandle方法

    • 此时若将其返回值改true,代码如下

    package at.guigu.interceptor;import org.springframework.stereotype.Component;
    import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;@Component
    public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");return true;}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也以返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
    }
    

    此时的运行截图如下

    在这里插入图片描述

配置类形式方法二

  • 此处配置类形式方法一中可以将第二、三步合并,即:让SpringMVC的核心配置类SpringMvcConfiguration继承WebMvcConfigurer接口并重写其中的addInterceptors方法来进行拦截器的路径映射配置,步骤如下

    • Step1: 创建一个与三层架构包同级的interceptor包并在该包下创建一个实现HandlerInterceptor接口的拦截器类,代码略

    • Step2: 让SpringMVC的核心配置类SpringMvcConfiguration继承WebMvcConfigurer接口并重写其中的addInterceptors方法来进行拦截器的路径映射配置,代码如下:

      package at.guigu.config;import at.guigu.interceptor.MyInterceptor;
      import org.springframework.beans.factory.annotation.Autowired;
      import org.springframework.context.annotation.Bean;
      import org.springframework.context.annotation.ComponentScan;
      import org.springframework.context.annotation.Configuration;
      import org.springframework.web.servlet.config.annotation.EnableWebMvc;
      import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
      import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
      import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
      import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
      @Configuration
      // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
      // 加载controller对应的bean
      @ComponentScan("at.guigu.controller")
      // 自动配置 Spring MVC 的各种特性
      @EnableWebMvc
      public class SpringMvcConfiguration implements WebMvcConfigurer {// 依赖注入拦截器的bean@Autowiredprivate MyInterceptor myInterceptor;// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}// 配置静态资源路径@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {// 配置静态资源路径映射registry.addResourceHandler("/xxx/**").addResourceLocations("/xxx/");}// 配置拦截器路径@Overridepublic void addInterceptors(InterceptorRegistry registry) {//addInterceptor定义拦截器的具体实现类//addPathPatterns的参数代表拦截所有请求路径registry.addInterceptor(myInterceptor).addPathPatterns("/**");}
      }
      

      运行截图略

HandlerInterceptor接口详解

  • HandlerInterceptor接口方法

    HandlerInterceptor接口方法解释
    default boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception在请求到达Controller之前执行(即在在原始方法(即目标方法)执行之前执行),用于执行预处理逻辑。
    default void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception在Controller处理完请求之后,视图渲染之前执行(即在原始方法(目标方法)执行之后,视图对象返回之前执行)
    default void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception在整个请求完成之后执行(即在原始方法执行完并且视图对象也已返回之后执行)。注意:即使在Controller或视图渲染过程中抛出异常,该方法也会执行
  • 方法对应参数及返回值详解(相同参数不在重复解释)

    • preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)——前置处理方法
      • HttpServletRequest :请求对象
      • HttpServletResponse :响应对象
      • handler :被调用的处理器对象,本质上是一个方法对象,对反射技术中的Method对象进行了再包装
      • return true :表示请求可以交给Controller来执行原始方法。(当为拦截链环境时,会将请求交给下一个Interceptor的前置处理方法,直到所有前置处理方法均通过后,会交给Controller来执行原始方法)
      • return false :表示请求被拦截,Controller控制器和拦截链上后续的Interceptor都不会继续执行
    • postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView)——后置处理方法
      • ModelAndView :如果处理器执行完成具有返回结果,可以读取到对应数据与页面信息,并进行跳转
      • 调用该方法的前提是前置处理方法的返回值为true
    • afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex)——完成后处理方法
      • Exception : 如果处理器执行过程中出现异常对象,可以针对异常情况进行单独处理
      • 调用该方法的前提是前置处理方法的返回值为true
      • 无论处理器方法内部是否出现异常,该方法均会执行
  • preHandle方法经常用于执行预处理逻辑,比如:

    • 验证用户身份或权限。

    • 检查请求参数是否合法。

    • 记录日志或统计请求次数。

    • 当返回值为false时请求被拦截,后续逻辑(如Controller处理或postHandleafterCompletion方法)不会执行(可详见快速入门)。此时可以直接设置响应,例如返回一个错误状态码或转发或重定向到其他页面。示例如下

      • 在web项目核心目录(即webapp)下的jsp文件夹中创建error.jsp页面,代码如下,用于拦截示例

        <%@ page contentType="text/html;charset=UTF-8" language="java" %>
        <html><head><title>Title</title></head><body><h1>Error...</h1></body>
        </html>
        
      • MyInterceptor类的代码更改如下

        package at.guigu.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
        import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
        import javax.servlet.http.HttpServletResponse;public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");String str = request.getParameter("param");if ("yes".equals(str)) {return true;} else {request.getRequestDispatcher("/jsp/error.jsp").forward(request, response);return false;}}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也以返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
        }
        

        当param请求参数值不是yes

        在这里插入图片描述

        当param请求参数值是yes

        在这里插入图片描述

  • postHandle方法用于对ModelAndView进行修改,例如:

    • 添加全局数据到视图中(如页面标题、用户信息)。

    • 根据Controller的处理结果动态调整视图数据。

    • 注意:如果Controller方法没有返回ModelAndView(如使用@ResponseBodyRestController),则此方法可能不会被调用。

    • MyInterceptor类的代码更改如下

      package at.guigu.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
      import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;public class MyInterceptor implements HandlerInterceptor {// 在原始方法(即目标方法)执行之前执行@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running...");String str = request.getParameter("param");if ("yes".equals(str)) {return true;} else {request.getRequestDispatcher("/jsp/error.jsp").forward(request, response);return false;}}// 在原始方法(即目标方法)执行之后,视图对象返回之前执行@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {modelAndView.addObject("username", "李四");System.out.println("postHandle Running...");}// 在整个请求完成之后执行(即在原始方法执行完并且视图对象也以返回之后执行)@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running...");}
      }
      

      在这里插入图片描述

拦截器配置详解

  • 在快速入门中配置的是单个拦截器,在SpringMVC的核心配置文件中对应的代码如下

    <!--配置拦截器-->
    <mvc:interceptors><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptor"/></mvc:interceptor>
    </mvc:interceptors>
    
  • 用到的标签

    标签解释
    <mvc:interceptors>配置拦截器
    <mvc:interceptors>内嵌标签解释
    <mvc:interceptor>配置单个拦截器
    <mvc:interceptor>内嵌标签解释
    <mvc:mapping path/>定义拦截器的作用范围。当path属性为/**时,代表拦截所有请求路径;当为/xxx/**时,代表拦截xxx下的所有资源
    <bean class/>定义拦截器的具体实现类,class属性值为对应实现类的全限定名

拦截器链

  • 特点

    • 拦截器链执行顺序以拦截器添加顺序为准
    • 当拦截器中出现对原始处理器的拦截时,后续的拦截器均终止运行
    • 当拦截器运行中段时,此时仅运行配置在前面的拦截器的afterCompletion(即完成后处理方法)操作

    在这里插入图片描述

配置多个拦截器(拦截器链)步骤(配置文件形式)

环境准备等工作可见快速入门,此处以配置两个拦截器为例

  • Step1:interceptor包下创建第二个实现HandlerInterceptor接口的拦截器类,并重写其中的方法,代码如下

    package at.guigu.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;public class MyInterceptorTwo implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running222...");return true;}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running222...");}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running222...");}
    }
    
  • Step2:

  • 在SpringMVC的核心配置文件中配置拦截器

    <!--配置拦截器-->
    <mvc:interceptors><!--配置第一个拦截器实现类--><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptor"/></mvc:interceptor><!--配置第二个拦截器实现类--><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptorTwo"/></mvc:interceptor>
    </mvc:interceptors>
    
    • SpringMVC核心配置文件完整代码如下

      <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--mvc的注解驱动--><mvc:annotation-driven/><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的视图名称前缀属性prefix的值设为/jsp/--><property name="prefix" value="/jsp/"></property><!--将InternalResourceViewResolver类中的视图名称后缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--配置拦截器--><mvc:interceptors><!--配置第一个拦截器实现类--><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptor"/></mvc:interceptor><!--配置第二个拦截器实现类--><mvc:interceptor><!--定义拦截器的作用范围:此处表示拦截所有请求路径--><mvc:mapping path="/**"/><!--定义拦截器的具体实现类,class属性值为对应实现类的全限定名--><bean class="at.guigu.interceptor.MyInterceptorTwo"/></mvc:interceptor></mvc:interceptors>
      </beans>
      

      在这里插入图片描述

配置多个拦截器(拦截器链)步骤(配置类形式)

环境准备等工作可见快速入门,此处以配置两个拦截器为例

  • Step1:interceptor包下创建第二个实现HandlerInterceptor接口的拦截器类,并重写其中的方法,代码如下

    package at.guigu.interceptor;import org.springframework.stereotype.Component;
    import org.springframework.web.servlet.HandlerInterceptor;
    import org.springframework.web.servlet.ModelAndView;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    @Component
    public class MyInterceptorTwo implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle Running222...");return true;}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println("postHandle Running222...");}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion Running222...");}
    }
    
  • Step2: 在config包下的继承WebMvcConfigurationSupport类的子类SpringMvcSupport中依赖注入MyInterceptorTwo的bean,代码如下:

    package at.guigu.config;import at.guigu.interceptor.MyInterceptor;
    import at.guigu.interceptor.MyInterceptorTwo;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
    import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
    import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;// 配置静态资源以及拦截器的路径映射
    @Configuration
    public class SpringMvcSupport extends WebMvcConfigurationSupport {@Autowiredprivate MyInterceptor myInterceptor;@Autowiredprivate MyInterceptorTwo myInterceptorTwo;// 配置静态资源路径@Overrideprotected void addResourceHandlers (ResourceHandlerRegistry registry) {// 配置静态资源路径映射registry.addResourceHandler("/xxx/**").addResourceLocations("/xxx/");}// 配置拦截器路径@Overrideprotected void addInterceptors(InterceptorRegistry registry) {//addInterceptor定义拦截器的具体实现类//addPathPatterns的参数代表拦截所有请求路径registry.addInterceptor(myInterceptor).addPathPatterns("/**");registry.addInterceptor(myInterceptorTwo).addPathPatterns("/**");}
    }
    
  • Step3: 在SpringMVC的核心配置类中利用@ComponentScan注解扫描拦截器类MyInterceptor以及SpringMvcSupport类所在包,代码如下

    package at.guigu.config;import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.ComponentScan;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.web.servlet.config.annotation.EnableWebMvc;
    import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
    @Configuration
    // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
    // 加载controller对应的bean
    @ComponentScan({"at.guigu.controller", "at.guigu.interceptor", "at.guigu.config"})
    // 自动配置 Spring MVC 的各种特性
    @EnableWebMvc
    public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}
    }
    

    在这里插入图片描述

SpringMVC异常处理

  • 异常常见位置与诱因

    • 框架内部抛出的异常:因使用不合规导致
    • 数据层抛出的异常:因外部服务器故障导致(例如:服务器访问超时)
    • 业务层抛出的异常:因业务逻辑书写错误导致(例如:遍历业务书写操作,导致索引异常等)
    • 表现层抛出的异常:因数据收集、校验等规则导致(例如:不匹配的数据类型间导致异常)
    • 工具类抛出的异常:因工具类书写不严谨不够健壮导致(例如:必要释放的连接长期未释放等)
  • 项目异常分类及处理方案

    项目异常分类场景处理步骤
    业务异常(BusinessException)1.规范的用户行为产生的异常;2.不规范的用户行为操作产生的异常发送对应消息传递给用户,提醒规范操作
    系统异常(SystemException)项目运行过程中可预计且无法避免的异常1.记录日志;2.发送固定消息传递给用户,安抚用户;3.发送特定消息给运维人员,提醒维护;
    其他异常(Exception)编程人员未预期到的异常1.发送固定消息传递给用户,安抚用户;2.发送特定消息给编程人员,提醒维护(纳入预期范围内)3.记录日志
  • 系统异常分类

    系统异常(SystemException)分类解决方式
    预期异常通过捕获异常获取异常信息
    运行时异常RuntimeException通过规范代码开发、测试等手段减少运行时异常发生
  • 异常处理思路

    在ssm框架中,各个层级出现的所有均异常向上抛出到表现层进行处理,持久层抛给业务层,业务层抛给表现层,表现层继续向上抛出给SpringMVC的核心前端控制器,最后由SpringMVC的核心前端控制器交由异常处理器进行异常处理

    在这里插入图片描述

  • 异常处理器

    • 底层使用的是AOP的思想,它能够集中统一的处理项目中出现的异常

    在这里插入图片描述

  • 异常处理的方式

    • 简单异常处理:使用SpringMVC提供的简单异常处理器SimpleMappingExceptionResolver
    • 自定义异常处理:
      • 方式一:实现SpringMVC的异常处理接口HandlerExceptionResolver来自定义自己的异常处理器
      • 方式二:
  • 自定义异常的方式

    自定义异常方式使用场景
    创建继承RuntimeException(运行时异常)的子类程序内部的错误或不可恢复的错误,不强制要求捕获或声明
    创建继承Exception(可检查异常)的子类用于可恢复的错误或外部因素引起的错误,需要强制要求捕获或声明

    注意:

    ​ 1.在本示例中并没有分太清,只是为了演示作用,后续真实项目可根据情况选择;

    ​ 2.在配置文件形式的示例中创建继承Exception(可检查异常)的子类来创建自定义的异常;

    ​ 3.在配置类形式的示例中创建继承RuntimeException(运行时异常)的子类来创建自定义的异常。

    ​ 4.通过创建继承Exception(可检查异常)的子类来创建自定义的异常在示例中只进行了简写,并未完整给出具体操作代码,而在创建继承RuntimeException(运行时异常)的子类来创建自定义的异常中进行了详细的书写,两者是一样的,示例中仅以一个为例进行精写

异常处理的两种方式(配置文件形式)

环境准备

  • Step1: 导入坐标(略,可详见快速入门)

  • Step2: 右键源代码配置文件目录(即资源文件resources)→NewXML Configuration FileSpring Config,创建Spring和SpringMVC的核心配置文件,代码截图如下

    在这里插入图片描述

  • Step3: 创建业务层service包、controller包、异常exception包,代码截图如下

    • 注意:在service包中模拟各种异常来进行后续的示例操作

    在这里插入图片描述

  • Step4: 配置web项目核心目录(即webapp)下的WEB-INF中的web.xml,并在webapp目录下创建jsp文件夹,并写入4个页面success.jsp、error.jsp、errorshow1.jsp、errorshow5.jsp,代码截图如下

    • 注意:success.jsp为无异常时前端显示页面;error.jsp为默认错误视图;errorshow1.jsp、errorshow5.jsp分别为业务层show1()show2()方法抛出异常后所映射的视图

    在这里插入图片描述

  • 此时运行后前端页面会报对应异常,分别如下图所示

    在这里插入图片描述

    在这里插入图片描述

    在这里插入图片描述

    在这里插入图片描述

    在这里插入图片描述

  • 初始项目结构如下

    在这里插入图片描述

简单异常的处理(配置文件形式)

  • 注意:SpringMVC已经定义好了该类型的转换器,在使用时可根据项目情况在SpringMVC的核心配置文件中进行相应 异常与视图 的映射配置

  • 简单异常处理只需要在SpringMVC核心配置文件中配置异常处理机制即可,代码如下

    <!--配置异常处理机制-->
    <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver"><property name="defaultErrorView" value="error1"/><property name="exceptionMappings"><map><entry key="at.guigu.exception.MyException" value="error2"/><entry key="java.lang.ClassCastException" value="error3"/></map></property>
    </bean>
    

    原理: 通过SpringMVC提供的简单异常处理器SimpleMappingExceptionResolver配置异常处理机制。当某个异常抛出时,Spring会根据配置的 exceptionMappings 映射到异常所对应的视图。如果抛出的异常没有在 exceptionMappings 中配置,那么会使用 defaultErrorView 指定的默认错误视图。

    示例代码解释: 当抛出的异常为MyException时则会映射到视图error2.jsp;当抛出异常为ClassCastException时则会映射到视图error3.jsp;若抛出的异常没有在 exceptionMappings 中配置,则此时按照defaultErrorView(即默认异常视图)映射到error1.jsp

    注意: 在配置异常处理机制的代码中,视图均为带后缀名.jsp,是因为已在SpringMVC的核心配置文件中提前配置了内部资源视图解析器

  • 代码运行示例

    • 未在SpringMVC核心配置文件中配置异常处理机制时,此时由于有异常所以前端会显示异常页面,可详见环境准备中的运行截图

    • 在SpringMVC核心配置文件中配置异常处理机制,完整代码如下

      <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--mvc的注解驱动--><mvc:annotation-driven/><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的视图名称前缀属性prefix的值设为/jsp/--><property name="prefix" value="/jsp/"></property><!--将InternalResourceViewResolver类中的视图名称后缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--配置异常处理机制--><bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver"><property name="defaultErrorView" value="error"/><property name="exceptionMappings"><map><entry key="java.lang.ClassCastException" value="errorShow1"/><entry key="at.guigu.exception.MyException" value="errorshow5"/></map></property></bean>
      </beans>
      
    • 运行截图如下

      • show1() ClassCastException异常自动映射到errorShow1.jsp页面

        在这里插入图片描述

      • show5() 自定义MyException异常自动映射到errorShow5.jsp页面

        在这里插入图片描述

      • 其它异常自动映射到error.jsp页面

        在这里插入图片描述

自定义异常的处理方式一(配置文件及配置类形式)

  • 步骤

    • exception包下创建继承Exception(可检查异常)的子类来创建自定义的异常
    • 创建实现HandlerExceptionResolver接口的异常处理器类
    • 在SpingMVC的核心配置文件或配置类中配置自定义异常处理机制
    • 编写异常映射页面
    • 测试异常跳转
  • 代码实现

    • Step1: 创建继承Exception(可检查异常)的子类来创建自定义的异常

      • 本步骤已在环境准备中完成,此处省略
    • Step2: 创建一个与三层架构包同级的resolver包,并在该包下创建一个实现HandlerExceptionResolver接口的异常处理器类MyExceptionResolver,并重写其中的resolveException方法,代码如下

      package at.guigu.resolver;import at.guigu.exception.MyException;
      import org.springframework.web.servlet.HandlerExceptionResolver;
      import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;public class MyExceptionResolver implements HandlerExceptionResolver {/**** @param httpServletRequest* @param httpServletResponse* @param o* @param e:为抛出的异常对象* @return :返回ModelAndView对象,为异常映射的视图*/@Overridepublic ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {ModelAndView modelAndView = new ModelAndView();// 若是ClassCastException异常则映射到errorShow1.jsp页面if (e instanceof ClassCastException) {modelAndView.setViewName("errorShow1");// 若是自定义MyException异常则映射到errorShow5.jsp页面} else if (e instanceof MyException) {modelAndView.setViewName("errorShow5");// 若是其它异常则映射到error.jsp页面} else {modelAndView.setViewName("error");}return  modelAndView;}
      }
      
    • Step3: 在SpingMVC的核心配置文件或配置类中配置自定义异常处理机制,代码如下:

      • SpingMVC的核心配置文件

        <?xml version="1.0" encoding="UTF-8"?>
        <beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--mvc的注解驱动--><mvc:annotation-driven/><!--配置Controller层的注解的组件扫描--><context:component-scan base-package="at.guigu.controller"></context:component-scan><!--等同于<context:component-scan base-package="at.guigu">type指定要扫描的内容为注解,expression指定要扫描的对应注解的全限定名只扫描at.guigu包下有@Controller注解的类<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/></context:component-scan>--><!--配置内部资源视图解析器--><bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"><!--将InternalResourceViewResolver类中的视图名称前缀属性prefix的值设为/jsp/--><property name="prefix" value="/jsp/"></property><!--将InternalResourceViewResolver类中的视图名称后缀属性suffix的值设为.jsp--><property name="suffix" value=".jsp"></property></bean><!--配置静态资源的路径映射,让 Spring MVC 可以处理静态文件--><mvc:default-servlet-handler/><!--配置异常处理机制<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver"><property name="defaultErrorView" value="error"/><property name="exceptionMappings"><map><entry key="java.lang.ClassCastException" value="errorShow1"/><entry key="at.guigu.exception.MyException" value="errorShow5"/></map></property></bean>--><!--配置自定义异常处理机制--><bean class="at.guigu.resolver.MyExceptionResolver"/>
        </beans>
        
      • SpingMVC的核心配置类

        package at.guigu.config;import at.guigu.resolver.MyExceptionResolver;
        import org.springframework.context.annotation.Bean;
        import org.springframework.context.annotation.ComponentScan;
        import org.springframework.context.annotation.Configuration;
        import org.springframework.context.annotation.Import;
        import org.springframework.web.servlet.config.annotation.EnableWebMvc;
        import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
        @Configuration
        // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller"></context:component-scan>
        // 加载controller对应的bean
        @ComponentScan(basePackages = {"at.guigu.controller"})
        // 自动配置 Spring MVC 的各种特性,比如:类型转换器、mvc的注解驱动<mvc:annotation-driven/>
        @EnableWebMvc
        // 引入配置静态资源类
        @Import(SpringMvcSupport.class)
        public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}// 配置自定义异常处理器@Beanpublic MyExceptionResolver myExceptionResolver() {return new MyExceptionResolver();}
        }
        
    • Step4: 编写异常映射页面。(本步骤已在环境准备工作中实现,此处省略)

    • 运行截图如下

      • show1() ClassCastException异常自动映射到errorShow1.jsp页面

        在这里插入图片描述

      • show5() 自定义MyException异常自动映射到errorShow5.jsp页面

        在这里插入图片描述

      • 其它异常自动映射到error.jsp页面

        在这里插入图片描述

异常处理的两种方式(配置类形式)

  • 方式一简单异常处理:使用SpringMVC提供的简单异常处理器SimpleMappingExceptionResolver

  • 方式二自定义异常的处理:实现Spring的异常处理接口HandlerExceptionResolver自定义自己的异常处理器

  • 用到的注解如下

    类注解解释
    @ControllerAdvice定义全局的异常处理、数据绑定、全局模型属性等功能,用于集中处理多个控制器中的异常、数据绑定、模型属性等
    @RestControllerAdvice定义全局的异常处理、数据绑定、全局模型属性等功能,用于集中处理多个控制器中的异常、数据绑定、模型属性等
    方法注解解释
    @ExceptionHandler设置指定异常的处理方案,出现异常后终止原始控制器执行,并转入当前方法执行

环境准备

  • Step1: 导入坐标(略,可详见快速入门)

  • Step2: 创建config包,并在该包下创建Spring核心配置类SpringConfiguration,SpringMVC核心配置类SpringMvcConfiguration、配置静态资源的继承WebMvcConfigurationSupport类的子类SpringMvcSupport、代替web.xml文件的继承AbstractAnnotationConfigDispatcherServletInitializer的子类ServletConfiguration,如图所示

    在这里插入图片描述

  • Step3: 创建业务层service包、controller包、异常exception包,代码截图如下

    • 注意:在service包中模拟各种异常来进行后续的示例操作

      在这里插入图片描述

  • Step4: 在webapp目录下创建jsp文件夹,并写入4个页面success.jsp、error.jsp、errorshow1.jsp、errorshow5.jsp,代码截图如下

    • 注意:success.jsp为无异常时前端显示页面;error.jsp为默认错误视图;errorshow1.jsp、errorshow5.jsp分别为业务层show1()show2()方法抛出异常后所映射的视图

    在这里插入图片描述

  • 此时运行后前端页面会报对应异常,分别如下图所示

    在这里插入图片描述

    在这里插入图片描述

    在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

简单异常的处理(配置类形式)

  • Step1: 在controller包下创建ProjectEcepAdvice类,代码如下

    • Step1-1: 添加@ControllerAdvice注解或@RestControllerAdvice注解
      • 若当前controller包下使用的是@Controller注解则为ProjectEcepAdvice类添加@ControllerAdvice注解
      • 若当前controller包下使用的是@RestController注解则为ProjectEcepAdvice类添加@RestControllerAdvice注解
    • Step1-2: 写一个处理异常的方法doException,并将拦截的异常作为参数传入
    • Step1-3: 给该方法添加一个@ExceptionHandler(value)注解,并设置value属性,以此来指定捕获的异常类型
      • 捕获所有类型异常:@ExceptionHandler(Exception.class)
      • 捕获指定类型异常:@ExceptionHandler({FileNotFoundException.class, IOException.class})
    package at.guigu.controller;import at.guigu.exception.MyException;
    import org.springframework.web.bind.annotation.ControllerAdvice;
    import org.springframework.web.bind.annotation.ExceptionHandler;
    import org.springframework.web.servlet.ModelAndView;@ControllerAdvice
    public class ProjectEcepAdvice {// 捕获所有异常进行处理@ExceptionHandler(Exception.class)public ModelAndView doException(Exception e){ModelAndView modelAndView = new ModelAndView();// 若是ClassCastException异常则映射到errorShow1.jsp页面if (e instanceof ClassCastException) {modelAndView.setViewName("errorShow1");// 若是自定义MyException异常则映射到errorShow5.jsp页面} else if (e instanceof MyException) {modelAndView.setViewName("errorShow5");// 若是其它异常则映射到error.jsp页面} else {modelAndView.setViewName("error");}return  modelAndView;}
    }
    
  • 运行截图如下

    • show1() ClassCastException异常自动映射到errorShow1.jsp页面

      在这里插入图片描述

    • show5() 自定义MyException异常自动映射到errorShow5.jsp页面

      在这里插入图片描述

    • 其它异常自动映射到error.jsp页面

      在这里插入图片描述

  • 注意:若ProjectEcepAdvice类并未在controller包下,而是在其它包下,则此时需要在SpringMVC的核心配置类中用@ComponentScan注解引入,比如:

    • 此时我并未在controller包下创建该类,而是创建了一个与三层架构包同级的advice包,并在该包下创建ProjectEcepAdvice类,此时就需要SpringMVC的核心配置类中利用@ComponentScan注解引入该包,让其能够扫描到,代码如下:

      package at.guigu.config;import org.springframework.context.annotation.Bean;
      import org.springframework.context.annotation.ComponentScan;
      import org.springframework.context.annotation.Configuration;
      import org.springframework.context.annotation.Import;
      import org.springframework.web.multipart.commons.CommonsMultipartResolver;
      import org.springframework.web.servlet.config.annotation.EnableWebMvc;
      import org.springframework.web.servlet.view.InternalResourceViewResolver;// 该注解代表该类是SpringMVC的核心配置类
      @Configuration
      // 配置注解的组件扫描<context:component-scan base-package="at.guigu.controller, at.guigu.advice"></context:component-scan>
      // 加载controller对应的bean
      @ComponentScan(basePackages = {"at.guigu.controller", "at.guigu.advice"})
      // 自动配置 Spring MVC 的各种特性,比如:类型转换器、mvc的注解驱动<mvc:annotation-driven/>
      //@EnableWebMvc
      // 引入配置静态资源类
      @Import(SpringMvcSupport.class)
      public class SpringMvcConfiguration {// 配置视图解析器@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/jsp/");  // 设置视图文件路径前缀resolver.setSuffix(".jsp");   // 设置视图文件后缀return resolver;}
      }
      

自定义异常的处理(配置类形式)

  • 步骤

    • exception包下创建继承RuntimeException的子类SystemException来自定义一个异常
    • 在SpingMVC的核心配置文件中配置自定义异常处理机制
    • 编写异常映射页面
    • 测试异常跳转
  • 代码实现

    • Step1: 创建一个与三层架构包同级的exception包,并在该包下创建继承RuntimeException的子类SystemException自定义一个系统异常 ,代码如下:

      • 设置一个code私有属性来接收异常对应的编号
      • 添加构造器(注意:RuntimeException有5个构造器,可根据需要选用一个或多个)
      • 添加getset方法
      package at.guigu.exception;public class SystemException extends RuntimeException {// 给异常添加编号private Integer code;// 添加构造方法public SystemException(Integer code, String message) {super(message);this.code = code;}public SystemException(Integer code, String message, Throwable cause) {super(message, cause);this.code = code;}public Integer getCode() {return code;}public void setCode(Integer code) {this.code = code;}
      }
      
    • Step2: 按照第一步在创建一个BusinessException类来自定义一个业务异常 ,代码如下:

      package at.guigu.exception;public class BusinessException extends RuntimeException {// 给异常添加编号private Integer code;// 添加构造方法public BusinessException(Integer code, String message) {super(message);this.code = code;}public BusinessException(Integer code, String message, Throwable cause) {super(message, cause);this.code = code;}public Integer getCode() {return code;}public void setCode(Integer code) {this.code = code;}
      }
      
    • Step3: 在controller包下创建Code类,来封装自定义异常的响应码,代码如下

      package at.guigu.controller;public class Code {// 50001代表自定义异常BusinessExceptionpublic static final Integer BUSS_ERR = 50001;// 50002代表自定义异常SystemExceptionpublic static final Integer SYST_ERR = 50002;// 50003代表自定义MyException异常public static final Integer MYEXECP_ERR = 50003;// 50004代表除自定义异常外的其它异常public static final Integer OTHER_ERR = 50004;
      }
      
    • Step4: 将业务层中可能出现的异常进行拦截包装,然年将其转换成自定义的异常,代码如下:

      • 此处只进行部分演示
        package at.guigu.service;import at.guigu.controller.Code;import at.guigu.exception.MyException;import at.guigu.exception.BusinessException;import at.guigu.exception.SystemException;import org.springframework.stereotype.Service;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.InputStream;@Servicepublic class UserService {public void show1() {// 将show1()抛出的类型转换异常拦截包装并将其转换成自定义的异常Object str = "zhangsan";if (str == "zhangsan") {throw new BusinessException(Code.BUSS_ERR, "抛出类型转换异常");}}public void show2() {// 将show2()抛出的除0异常拦截包装并将其转换成自定义的异常try {int i = 1/0;} catch (Exception e) {throw new SystemException(Code.SYST_ERR, "抛出除0异常,请重新输入除数", e);}}public void show3() throws FileNotFoundException {System.out.println("抛出文件找不到异常");InputStream inputStream = new FileInputStream("C:/xxx/xxx/xxx.txt");}public void show4() {System.out.println("抛出空指针异常");String str = null;str.length();}public void show5() throws MyException {System.out.println("抛出自定义异常");throw new MyException();}}
      
    • Step6: 在controller包下创建ProjectEcepAdvice类,代码如下

      • Step6-1: 添加@ControllerAdvice注解或@RestControllerAdvice注解
        • 若当前controller包下使用的是@Controller注解则为ProjectEcepAdvice类添加@ControllerAdvice注解
        • 若当前controller包下使用的是@RestController注解则为ProjectEcepAdvice类添加@RestControllerAdvice注解
      • Step6-2: 写一个处理异常的方法doException,并将拦截的异常作为参数传入
      • Step6-3: 给该方法添加一个@ExceptionHandler(value)注解,并设置value属性,以此来指定捕获的异常类型
        • 捕获所有类型异常:@ExceptionHandler(Exception.class)
        • 捕获指定类型异常:@ExceptionHandler({FileNotFoundException.class, IOException.class})
      package at.guigu.controller;import at.guigu.exception.BusinessException;
      import at.guigu.exception.MyException;
      import at.guigu.exception.SystemException;
      import org.springframework.web.bind.annotation.ControllerAdvice;
      import org.springframework.web.bind.annotation.ExceptionHandler;
      import org.springframework.web.servlet.ModelAndView;import java.io.FileNotFoundException;@ControllerAdvice
      public class ProjectEcepAdvice {// 捕获指定的自定义异常BusinessException进行处理@ExceptionHandler(BusinessException.class)public ModelAndView doBusinessException(BusinessException e){ModelAndView modelAndView = new ModelAndView();modelAndView.setViewName("errorShow1");// 发送对应消息传递给用户,提醒规范操作return modelAndView;}// 捕获指定的自定义异常SystemException进行处理@ExceptionHandler(SystemException.class)public ModelAndView doSystemException(SystemException e){// 记录日志// 发送消息给运维// 发送对应消息传递给用户// 发送邮件以及异常发送给开发人员ModelAndView modelAndView = new ModelAndView();modelAndView.setViewName("error");return modelAndView;}// 捕获除自定义异常MyException进行处理@ExceptionHandler(MyException.class)public ModelAndView doMyException(MyException e){ModelAndView modelAndView = new ModelAndView();modelAndView.setViewName("errorShow5");return modelAndView;}// 捕获除自定义外的其它异常进行处理@ExceptionHandler({FileNotFoundException.class, NullPointerException.class})public ModelAndView doException(Exception e){// 记录日志// 发送消息给运维// 发送对应消息传递给用户// 发送邮件以及异常发送给开发人员ModelAndView modelAndView = new ModelAndView();modelAndView.setViewName("error");return modelAndView;}
      }
      

      运行截图略

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/63474.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PDF拆分之怎么对批量的PDF文件进行分割-免费PDF编辑工具分享

>>更多PDF文件处理应用技巧请前往 96缔盟PDF处理器 主页 查阅&#xff01; ——————————————————————————————————————— 当然了&#xff0c;单个文件或者其他任意的文件个数的拆分也是支持的&#xff01; 序言 我之前的文章也有…

EmoAva:首个大规模、高质量的文本到3D表情映射数据集。

2024-12-03&#xff0c;由哈尔滨工业大学&#xff08;深圳&#xff09;的计算机科学系联合澳门大学、新加坡南洋理工大学等机构创建了EmoAva数据集&#xff0c;这是首个大规模、高质量的文本到3D表情映射数据集&#xff0c;对于推动情感丰富的3D头像生成技术的发展具有重要意义…

【开源免费】基于Vue和SpringBoot的课程答疑系统(附论文)

博主说明&#xff1a;本文项目编号 T 070 &#xff0c;文末自助获取源码 \color{red}{T070&#xff0c;文末自助获取源码} T070&#xff0c;文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析…

Spring Boot 整合 Druid 并开启监控

文章目录 1. 引言2. 添加依赖3. 配置数据源4. 开启监控功能5. 自定义 Druid 配置&#xff08;可选&#xff09;6. 访问监控页面7. 注意事项8. 总结 Druid 是一个由阿里巴巴开源的高性能数据库连接池&#xff0c;它不仅提供了高效的连接管理功能&#xff0c;还自带了强大的监控和…

第二十四周机器学习笔记:动手深度学习之——统计学习知识

第二十四周周报 摘要Abstract1.监督学习和无监督学习1.1 监督学习&#xff08;Supervised Learning&#xff09;1.2 无监督学习&#xff08;Unsupervised Learning&#xff09; 2.线性回归模型3.K-means聚类算法3.1 K-means算法的具体步骤&#xff1a; 4.决策树4.1 划分选择的目…

云计算vspere 安装过程

1 材料的准备 1 安装虚拟机 vmware workstation 2 安装esxi 主机 3 在esxi 主机上安装windows 2018 dns 服务器 4 在虚拟机上安装windows 2018 服务器 6 安装vcenter 5 登入界面测试 这里讲一下&#xff0c;由于部署vspere 需要在windows 2012 服务器上部…

【青牛科技】应用于音频信号处理系统的D258 是由两个独立的高增益运算放大器组成

概述&#xff1a; D258是由两个独立的高增益运算放大器组成。可以是单电源工作&#xff0c;也可以是双电源工作,电源的电流消耗与电源电压大小无关。应用范围包括变频放大器、DC增益部件和所有常规运算放大电路。 主要特点&#xff1a; ● 可单电源或双电源 工作 ● 在一个封…

HTML旋转爱心(完整代码)

目录 写在前面 完整代码 下载代码 代码分析 系列文章 写在后面 写在前面 HTML语言实现旋转爱心的完整代码。 完整代码 <!DOCTYPE html> <html lang="en"><head><title>Love</title><meta charset="utf-8">&l…

1.文本方块方法(Spacy Text Splitter 方法)Can‘t find model ‘zh_core_web_sm‘

一、概述 执行如下&#xff1a; def split_spacy(text):import spacynlp spacy.load( "zh_core_web_sm" ) doc nlp(text) for s in doc.sents: print(s) # d:\programdata\anaconda3\envs\python310\lib\site-packages if __name__"__main__":text &q…

maven高级管理

1. 依赖管理 pom.xml使用标签来进行依赖管理&#xff0c;具体涉及 依赖传递可选依赖排除依赖 依赖是具有传递性 **说明:**A代表自己的项目&#xff1b;B,C,D,E,F,G代表的是项目所依赖的jar包&#xff1b;D1和D2 E1和E2代表是相同jar包的不同版本 (1) A依赖了B和C,B和C有分别…

自建服务器,数据安全有保障

在远程桌面工具的选择上&#xff0c;向日葵和TeamViewer功能强大&#xff0c;但都存在收费昂贵、依赖第三方服务器、数据隐私难以完全掌控等问题。相比之下&#xff0c;RustDesk 凭借开源免费、自建服务的特性脱颖而出&#xff01;用户可以在自己的服务器上部署RustDesk服务端&…

一些常见网络安全术语

1、黑帽 为非法目的进行黑客攻击的人&#xff0c;通常是为了经济利益。他们进入安全网络以销毁&#xff0c;赎回&#xff0c;修改或窃取数据&#xff0c;或使网络无法用于授权用户。这个名字来源于这样一个事实&#xff1a;老式的黑白西部电影中的恶棍很容易被电影观众识别&…

Linux-PWM驱动实验

在裸机篇我们已经学习过了如何使用 I.MX6ULL 的 PWM 外设来实现 LCD 的背光调节&#xff0c;其实在 Linux 的 LCD 驱动实验我们也提到过 I.MX6ULL 的 PWM 背光调节&#xff0c;但是并没有专门的去讲解 PWM 部分&#xff0c;本章我们就来学习一下 Linux 下的 PWM 驱动开发。 PWM…

wordpress网站安装了Linux宝塔面板,限制IP地址访问网站,只能使用域名访问网站

一、Linux服务器安装Linux宝塔面板 这个步骤参考网上其他教程。 二、Linux宝塔面板部署wordpress网站 这个步骤参考网上其他教程&#xff0c;保证网站能够正常访问&#xff0c;并且使用Linux宝塔面板申请并部署了SSL证书&#xff0c;使用https协议默认443端口正常访问。 三…

Docker基础【windows环境】

课程内容来自尚硅谷3小时速通Docker教程 1. Docker简介 Docker 通过 Docker Hub 实现一行命令安装应用&#xff08;镜像&#xff09;【Nginx&#xff0c;Mysql等】&#xff0c;避免繁琐的部署操作。同时通过轻量级&#xff08;相对于虚拟机&#xff09;的容器化的思想&#x…

flinkSql 将流和表的互相转换

流——>表 方式一 方式二 方式一&#xff1a;写sql DataStreamSource<String> source env.socketTextStream("localhost", 8881); // 表名&#xff0c;流&#xff0c;字段名称 tableEnv.createTemporaryView("t_1",source&#xff0c;$("…

core Webapi jwt 认证

core cookie 验证 Web API Jwt 》》》》用户信息 namespace WebAPI001.Coms {public class Account{public string UserName { get; set; }public string UserPassword { get; set; }public string UserRole { get; set; }} }》》》获取jwt类 using Microsoft.AspNetCore.Mvc…

运输层4——TCP格式(重点!)

目录 一、TCP报文段格式 二、最大报文长度 MSS 一、TCP报文段格式 长度&#xff1a;前20个字节固定 后4n个字节&#xff08;报文段格式不固定&#xff09; 1、源端和目的端&#xff1a;各2个字节 作用&#xff1a;指明TCP链接的发送 2、序号 4字节 作用&#xff1…

Android显示系统(03)- OpenGL ES - GLSurfaceView的使用

Android显示系统&#xff08;02&#xff09;- OpenGL ES - 概述 Android显示系统&#xff08;03&#xff09;- OpenGL ES - GLSurfaceView的使用 Android显示系统&#xff08;04&#xff09;- OpenGL ES - Shader绘制三角形 Android显示系统&#xff08;05&#xff09;- OpenGL…

关于睡懒觉

我们经常听到一个词&#xff1a;睡懒觉。 我认为&#xff0c;睡懒觉这个词&#xff0c;是错误的。 人&#xff0c;是需要睡眠的&#xff0c;睡不够&#xff0c;就不会醒。睡够了&#xff0c;自然会醒&#xff0c;也不想继续睡。不信你试试&#xff0c;睡够了&#xff0c;你…