一、 第一个 注解的 SpringMVC 程序
所谓 SpringMVC 的注解式开发是指,在代码中通过对类与方法的注解,便可完成处理器在 springmvc 容器的注册。注解式开发是重点。
本测试程序主要完成功能:用户提交一个请求,服务端处理器在接收到这个请求后,给出一条欢迎信息,在响应页面中显示该信息。
新建 maven项目
引入 servlet1 依赖 和 SpringMVC 依赖
依赖:
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
插件:
<build>
<plugins>
<!-- 编码和编译和JDK版本 -->
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.1</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
</plugins>
</build>
注册中间调度器
-
中央调入器的全限定性类名
该中央调度器为一个 Servlet,名称为DispatcherServlet
。中央调度器的全限定性类名在导入的 Jar 文件 spring-webmvc-5.2.5.RELEASE.jar 的第一个包org.springframework.web.servlet
下可找到。 -
<load-on-startup/>
在<servlet/>中添加<load-on-startup/>的作用是,标记是否在Web服务器(这里是Tomcat)启动时会创建这个 Servlet 实例。即是否在 Web 服务器启动时调用执行该 Servlet 的 init()方法,而不是在真正访问时才创建。
它的值必须是一个整数。
➢ 当值大于等于 0 时,表示容器在启动时就加载并初始化这个 servlet,数值越小,该 Servlet的优先级就越高,其被创建的也就越早;
➢ 当值小于 0 或者没有指定时,则表示该 Servlet 在真正被使用时才会去创建。
➢ 当值相同时,容器会自己选择创建顺序 -
<url-pattern/>
对于<url-pattern/>,可以写为 / ,建议写为*.do
的形式。 -
配置文件位置与名称
注册完毕后,可直接在服务器上发布运行。
此时,访问浏览器页面,控制台均会抛出FileNotFoundException 异常
。即默认要从项目根下的 WEB-INF 目录下找名称为 Servlet 名称 -servlet.xml 的配置文件。这里的“Servlet 名称”指的是注册中央调度器<servlet-name/>标签中指定的 Servlet 的 name 值。本例配置文件名为springmvc-servlet.xm
l。
而一般情况下,配置文件是放在类路径下,即 resources 目录下。所以,在注册中央调度器时,还需要为中央调度器设置查找 SpringMVC 配置文件路径,及文件名。= 因此如果需要在服务器上发布后还能够访问,应当指定具体的根路径。代码如下:
-
DispatcherServlet的两个参数
打开 DispatcherServlet 的源码,其继承自FrameworkServlet
,而该类中有一个属性contextConfigLocation
,用于设置SpringMVC
配置文件的路径及文件名。该初始化参数的属性
就来自于这里。
创建 SpringMVC 配置文件
在工程的类路径即 src 目录下创建 SpringMVC 的配置文件 springmvc.xml
。该文件名可以
任意命名
创建处理器
在类上与方法上添加相应注解即可。
@Controller
:表示当前类为处理器@RequestMapping
:表示当前方法为处理器方法。该方法要对 value 属性所指定的 URI 进行处理与响应。被注解的方法的方法名可以随意
若有多个请求路径均可匹配该处理器方法的执行,则@RequestMapping
的 value 属性中可以写上一个数组。
ModelAndView
类中的addObject()
方法用于向其 Model 中添加数据。Model 的底层为一个 HashMap。- Model 中的数据存储在
request
作用域中,SringMVC 默认采用转发的方式跳转到视图,本次请求结束,模型中的数据被销毁。
声明组件扫描器
在 springmvc.xml
中注册组件扫描器
定义目标页面 jsp
在 webapp 目录下新建一个子目录 jsp,在其中新建一个 jsp 页面 show.jsp。
JSP 路径优化
修改视图解析器的注册
-
把 show.jsp 文件放到
/WEB-INF/jsp/
路径中 -
SpringMVC 框架为了避免对于请求资源路径与扩展名上的冗余,在视图解析器
I·nternalResouceViewResolver
中引入了请求的前辍与后辍。而 ModelAndView 中只需给出要跳
转页面的文件名即可,对于具体的文件路径与文件扩展名,视图解析器会自动完成拼接。配置前缀后缀如下:
修改处理器
这时,在处理器中直接使用 jsp 的名称即可,即 show
:
图解 SpringMVC 框架
- web 请求处理顺序
- SpringMVC 的 MVC 组件
spring MVC 的执行流程
- (1)浏览器提交请求到中央调度器
- (2)中央调度器直接将请求转给处理器映射器。
- (3)处理器映射器会根据请求,找到处理该请求的处理器,并将其封装为处理器执行链后返回给中央调度器。
- (4)中央调度器根据处理器执行链中的处理器,找到能够执行该处理器的处理器适配器。
- (5)处理器适配器调用执行处理器。
- (6)处理器将处理结果及要跳转的视图封装到一个对象 ModelAndView 中,并将其返回给处理器适配器 controller 的作用。
- (7)处理器适配器直接将结果返回给中央调度器。
- (8)中央调度器调用视图解析器,将 ModelAndView 中的视图名称封装为视图对象。
- (9)视图解析器将封装了的视图对象返回给中央调度器
- (10)中央调度器调用视图对象,让其自己进行渲染,即进行数据填充,形成响应对象。
- (11)中央调度器响应浏览器。
二、SpringMVC 注解详解
@RequestMapping 定义请求规则
指定模块名称
-
通过
@RequestMapping
注解可以定义处理器对于请求的映射规则。该注解可以注解在方法上,也可以注解在类上,但意义是不同的。value 属性值常以“/”开始。 -
@RequestMapping 的 value 属性用于定义所匹配请求的 URI。但对于注解在方法上与类上,其 value 属性所指定的 URI,意义是不同的。
-
一个
@Controller
所注解的类中,可以定义多个处理器方法。当然,不同的处理器方法所匹配的 URI 是不同的。这些不同的 URI 被指定在注解于方法之上的@RequestMapping 的
value 属性中。 -
但若这些请求具有相同的 URI 部分,则这些相同的 URI,可以被抽取到注解在类之上的@RequestMapping 的 value 属性中。
-
此时的这个 URI 表示模块的名称。==URI 的请求是相对于 Web 的根目录。==换个角度说,要访问处理器的指定方法,必须要在方法指定 URI 之前加上处理器类前定义的模块名称
-
举例:还是上面的入门项目中
修改处理器类 MyController。
则此时访问 other 和 some 界面的 url 就变成了test/other.jsp
和test/some.jsp
对请求提交方式的定义
对于 @RequestMapping
,其有一个属性method
,用于对被注解方法所处理请求的提交方式进行限制,即只有满足该 method 属性指定的提交方式的请求,才会执行该被注解方法。
- Method 属性的取值为 RequestMethod 枚举常量。常用的为
RequestMethod.GET
与
RequestMethod.POST
,分别表示提交方式的匹配规则为 GET 与 POST 提交。
- 客户端浏览器常用的请求方式如下:
- 也就是说,如果提交方式为 POST,那么请求发送的方式:要么使用表单请求,要么使用 AJAX 请求。其它请求方式被禁用。
- 当然,若不指定 method 属性,则无论是 GET 还是 POST 提交方式,均可匹配。即对于请求的提交方式无要求。
处理器方法的逐个参数接收
处理器方法可以包含以下四类参数,这些参数会在系统调用时由系统自动赋值,即程序员可在方法内直接使用。(即无需获取,可以直接使用)
➢ HttpServletRequest
➢ HttpServletResponse
➢ HttpSession
➢ 请求中所携带的请求参数
逐个参数接收
- 只要保证请求参数名与该请求处理方法的参数名相同即可。=
- 举例:
首先修改 index 页面
其次,在控制器层直接接收参数即可,只要名字相同,会自动装载
在 跳转的 jsp中,即 show.jsp 中,就可以直接获取对应的变量值
请求参数中文乱码问题
Spring 对于请求参数中的中文乱码问题,给出了专门的字符集过滤器:spring-web-5.2.5.RELEASE.jar
的org.springframework.web.filter
包下的 CharacterEncodingFilter
类。
解决方法
在 web.xml 中注册字符集过滤器,即可解决 Spring 的请求参数的中文乱码问题。
- 不过,最好将该过滤器注册在其它过滤器之前。因为过滤器的执行是按照其注册顺序进行的。
直接在项目 web.xml 上进行修改即可:
源码分析
字符集设置核心方法:
校正请求参数名 @RequestParam
所谓校正请求参数名,是指若请求 URL 所携带的参数名称与处理方法中指定的参数名不相同时,则需在处理方法参数前,添加一个注解@RequestParam(“请求参数名”),指定请求 URL 所携带参数的名称。(相当于别名的意思)
- 该注解是对处理器方法参数进行修饰的。value 属性指定请求参数的名称
举例
-
Step1:修改 index 页面
将表单中的参数名称修改的与原来不一样。
-
Step2:修改处理器类 MyController
-
required属性
@requestParam 属性中存在 一个 required 属性,如下所示:
对象参数接收
将处理器方法的参数定义为一个对象,只要保证请求参数名与这个对象的属性同名即可。也就是说和基本类型的操作方式一致。
举例:
- 在 controller 中定义student 并且加入hashmap中
- 直接在 jsp 中访问即可:
解读<url-pattern/>
配置详解
*.do
在没有特殊要求的情况下,SpringMVC 的中央调度器 DispatcherServlet 的<url-pattern/>常使用后辍匹配方式,如写为*.do 或者 *.action, *.mvc 等。/
可以写为/
,当时如果改为斜杠,DispatcherServlet 会将向静态资源的获取请求,例如.css、.js、.jpg、.png 等资源的获取请求,当作是一个普通的 Controller 请求。==中央调度器会调用处理器映射器为其查找相应的处理器。==当然也是找不到的,所以在这种情况下,所有的静态资源获取请求也均会报 404 错误。
举例:*.do 与 \ 的对比
需求:在 index.jsp 页面中存在一个访问图片的链接。该项目用于演示将 <url-pattern/>写为*.do
可以访问到该图片,而写为/
,则无法访问。
-
A、 在项目中添加图片
在项目的 WebRoot 下添加一个目录 images,并在其中添加一张图片资源。
-
B、 修改 index 页面
-
C、 修改<url-pattern/>的值
-
结果如下:
- 保持<url-pattern/>的值为
*.do
,扩展名方式,图片会正常显示。 - 将<url-pattern/>的值修改为
/
,则图片将无法显示。
- 保持<url-pattern/>的值为
<url-pattern/>的值并不是说写为/
后,静态资源就无法访问了。经过一些配置后,该问题也是可以解决的。
\ 下的静态资源访问
<mvc:default-servlet-handler/>
- 声明了<mvc:default-servlet-handler/>后,springmvc框架会在容器中创建
DefaultServletHttpRequestHandler
处理器对象。它会像一个检查员,对进入DispatcherServlet的URL进行筛查,如果发现是静态资源的请求,就将该请求转由Web应用服务器默认的Servlet处理。一般的服务器都有默认的Servlet。 - 在 Tomcat 中,有一个专门用于处理静态资源访问的 Servlet 名叫 DefaultServlet。其<servlet-name/>为 default。可以处理各种静态资源访问请求。该 Servlet 注册在 Tomcat 服务器的 web.xml 中。
- 具体步骤如下:
-
step1: 在 Tomcat 安装目录
/conf/web.xml。
-
step2: 在 springmvc.xml 中添加
<mvc:default-servlet-handler/>
标签
<mvc:default-servlet-handler/>表示使用 DefaultServletHttpRequestHandler 处理器对象。而该处理器调用了 Tomcat 的 DefaultServlet 来处理静态资源的访问请求。 -
step3: 当然了,要想使用
<mvc: …/>标签
,需要引入 mvc 约束
该约束可从 Spring 帮助文档中搜索关键字 spring-mvc.xsd 即可获取:docs/spring-framework-reference/htmlsingle/index.html
-
使用<mvc:resources/>(掌握)
- 在 Spring3.0 版本后,Spring 定义了专门用于处理静态资源访问请求的处理器
ResourceHttpRequestHandler。并且添加了<mvc:resources/>标签,专门用于解决静态资源无法访问问题。 - 需要在 springmvc 配置文件中添加如下形式的配置:
- location 表示静态资源所在目录。当然,目录不要使用
/WEB-INF/及其子目录
。 - mapping 表示对该资源的请求(以
/images/
开始的请求 ,如/image/beauty.jpg
,
/images/car.png
等)。注意,后面是两个星号**。
- location 表示静态资源所在目录。当然,目录不要使用
声明注解驱动
- 解决动态资源和静态资源冲突的问题,在 springmvc 配置文件加入:
三、 处理器方法的返回值
使用 @Controller
注解的处理器的处理器方法,其返回值常用的有四种类型:
➢ 第一种:ModelAndView
➢ 第二种:String
➢ 第三种:无返回值 void
➢ 第四种:返回自定义类型对象
根据不同的情况,使用不同的返回值。
返回 ModelAndView
- 若处理器方法处理完后,需要跳转到其它资源,且又要在跳转的资源间传递数据,此时处理器方法返回 ModelAndView 比较好。 当然,若要返回 ModelAndView,则处理器方法中需要定义 ModelAndView 对象。
- 在使用时,若该处理器方法只是进行跳转而不传递数据,或只是传递数据而并不向任何资源跳转(如对页面的 Ajax 异步响应),此时若返回 ModelAndView,则将总是有一部分多余:要么 Model 多余,要么 View 多余。即此时返回 ModelAndView 将不合适。
返回 String
处理器方法返回的字符串可以指定逻辑视图名,通过视图解析器解析可以将其转换为物理视图地址返回内部资源逻辑视图名
-
若要跳转的资源为内部资源,则视图解析器可以使用
InternalResourceViewResolver
内部资源视图解析器。此时处理器方法返回的字符串就是要跳转页面的文件名去掉文件扩展名后
的部分。这个字符串与视图解析器中的 prefix、suffix 相结合,即可形成要访问的 URI。
-
举例:直接在 controller 的方法中返回 “”show“”
-
当然,也可以直接返回资源的物理视图名(此时就不能配置前缀后缀)
返回 void(了解)
对于处理器方法返回 void 的应用场景,AJAX 响应.
- 若处理器对请求处理后,无需跳转到其它任何资源,此时可以让处理器方法返回 void。
例如,对于 AJAX 的异步请求的响应。
-
Step1:maven 加入 jackson 依赖
由于本项目中服务端向浏览器传回的是 JSON 数据,需要使用一个工具类将字符串包装为 JSON 格式,所以需要导入 JSON 的依赖。<dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>2.9.0</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>2.9.0</version> </dependency>
-
Step2:引入 jQuery 库
由于本项目要使用 jQuery 的 ajax()方法提交 AJAX 请求,所以项目中需要引入 jQuery 的库。在 WebRoot 下新建一个 Folder(文件夹),命名为 js,并将 jquery-1.11.1.js 文件放入其中。
并且在 index 页面中引入 JQuery 库文件
-
Step3:定义 index 页面
index 页面由两部分内容构成:一个是<button/>,用于提交 AJAX 请求;一个是<script/>,用于处理 AJAX 请求
点击按钮发起请求
-
Step4: 定义对象 Student
-
Step5:修改处理器类 MyController
处理器对于 AJAX 请求中所提交的参数,可以使用逐个接收的方式,也可以以对象的方式整体接收。只要保证 AJAX 请求参数与接收的对象类型属性同名。
以逐个方式接收参数:
返回 对象 Object
处理器方法也可以返回 Object 对象。这个 Object 可以是 Integer,String,自定义对象,Map,List 等、但返回的对象不是作为逻辑视图出现的,而是作为直接在页面显示的数据出现的。
- 返回对象,需要使用
@ResponseBody 注解
,将转换后的 JSON 数据放入到响应体中
。
因为需要使用到 JSON 数据,因此在进行返回之前,我们需要先对环境进行搭建
环境搭建
-
pom.xml 的修改: 由于返回 Object 数据,一般都是将数据转化为了 JSON 对象后传递给浏览器页面的。而这个由 Object 转换为 JSON,是由 Jackson 工具完成的。所以需要导入 Jackson 的相关 Jar 包。
依赖:<dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>2.9.0</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>2.9.0</version> </dependency>
-
声明注解驱动: 将 Object 数据转化为 JSON 数据,需要由消息转换器
HttpMessageConverter
完成。而转换器的开启,需要由<mvc:annotation-driven/>
来完成。SpringMVC 使用消息转换器实现请求数据和对象,处理器方法返回对象和响应输出之间
的自动转换-
当 Spring 容器进行初始化过程中,在
<mvc:annotation-driven/>
处创建注解驱动时,默认创建了七个 HttpMessageConverter 对象。
-
也就是说,我们注册
<mvc:annotation-driven/>
,就是为了让容器为我们创建 HttpMessageConverter 对象。
-
-
HttpMessageConverter 接口 :
HttpMessageConverter<T>是 Spring3.0 新添加的一个接口,负责将请求信息转换为一个对象(类型为 T),将对象(类型为 T)输出为响应信息。
HttpMessageConverter 接口定义的方法:boolean canRead(Class<?> clazz,MediaType mediaType):
指定转换器可以读取的对象类型,即转换器是否可将请求信息转换为 clazz 类型的对 象 , 同时指定支持MIME类 型(text/html,applaiction/json 等)boolean canWrite(Class<?> clazz,MediaType mediaType):
指定转换器是否可将 clazz 类型的对象写到响应流中,响应流支持的媒体类型在 MediaType 中定义。LIst<MediaType> getSupportMediaTypes():
该转换器支持的媒体类型。T read(Class<? extends T> clazz,HttpInputMessage inputMessage):
将请求信息流转换为 T 类型
的对象。void write(T t,MediaType contnetType,HttpOutputMessgae outputMessage):
将 T 类型的对象写到响应流中,同时指定相应的媒体类型为 contentType
-
加入注解驱动
<mvc:annotation-driven/>
后适配器类的 messageConverters 属性值分别如下所示。
返回自定义类型对象
返回自定义类型对象时,不能以对象的形式直接返回给客户端浏览器,而是将对象转换为 JSON 格式的数据发送给浏览器的。
- 由于转换器底层使用了Jackson转换方式将对象转换为JSON数据,所以需要导入Jackson的相关 Jar 包。
举例:
-
Step1:定义数据类
-
Step2:修改处理器 MyController
-
Step3:修改 index 页面
返回 List 集合
-
Step1:修改处理器 MyController
-
Step2:修改 index 页面
返回字符串对象
若要返回非中文字符串,将前面返回数值型数据的返回值直接修改为字符串即可。
- 但若返回的字符串中带有中文字符,则接收方页面将会出现乱码。此时需要使用
@RequestMapping
的produces 属性
指定字符集。 - produces 属性用于设置输出结果类型。
举例
-
直接修改处理器
-
修改页面
四、SpringMVC 核心技术
请求重定向与请求转发
当处理器对请求处理完毕后,向其它资源进行跳转时,有两种跳转方式:请求转发与重定向。而根据所要跳转的资源类型,又可分为两类:跳转到页面与跳转到其它处理器。
- 注意,对于请求转发的页面,可以是WEB-INF中页面;而重定向的页面,是不能为
WEB-INF
中页的。因为重定向相当于用户再次发出一次请求,而用户是不能直接访问 WEB-INF 中资源的。
SpringMVC 框架把原来 Servlet 中的请求转发和重定向操作进行了封装。现在可以使用简单的方式实现转发和重定向。 - forward:表示转发,实现
request.getRequestDispatcher("xx.jsp").forward()
- redirect:表示重定向,实现
response.sendRedirect("xxx.jsp")
SpringMVC 下的请求转发
处理器方法返回 ModelAndView 时,需在 setViewName()指定的视图前添加 forward:,且此时的视图不再与视图解析器一同工作,这样可以在配置了解析器时指定不同位置的视图。
- 视图页面必须写出相对于项目根的路径。forward 操作不需要视图解析器。处理器方法返回 String,在视图路径前面加入
forward: 视图完整路径
。
如下所示:
SpringMVC 重定向
在处理器方法返回的视图字符串前面加 redirect
,则可以实现重定向跳转
处理器方法定义:
异常处理
使用注解 @ExceptionHandler
可以将一个方法指定为异常处理方法。
- 该注解只有一个可选属性 value,为一个
Class<?>
数组,用于指定该注解的方法所要处理的异常类,即所要匹配的异常。 - 而被注解的方法,其返回值可以是
ModelAndView、String,或 void
,方法名随意,方法参数可以是Exception 及其子类对象、HttpServletRequest、HttpServletResponse 等。系统会自动为这些方法参数赋值。 - 对于异常处理注解的用法,也可以直接将异常处理方法注解于 Controller 之中。
举例: 接下来,我们将自定义三个异常并加入我们的 Controller 中。
自定义异常处理类
-
Step1: 自定义异常类
如下,我们定义了·三个异常类:NameException、AgeException、MyUserException
。其中MyUserException
是另外两个异常的父类。
-
Step2: 修改 Controller 抛出异常,即在处理器中定义一个处理异常的方法
-
Step3: 定义异常相应的方法
注意: 一般不这样使用。而是将异常处理方法专门定义在一个类中,作为全局的异常处理类。
定义全局异常处理类
- 需要使用注解
@ControllerAdvice
,字面理解就是“控制器增强”,是给控制器对象增强功能的。 使用@ControllerAdvice 修饰的类中可以使用@ExceptionHandler。 - 当使用@RequestMapping 注解修饰的方法抛出异常时,会执行
@ControllerAdvice
修饰的类中的异常处理方法。 - @ControllerAdvice 是使用@Component 注解修饰的,可以<context:component-scan>扫描到@ControllerAdvice 所在的类路径(包名),创建对象。
总结: 用 @ControllerAdvice
来修饰一个全局的异常类,然后类中每个方法用 @ExceptionHandler
进行注解。(value 值表示具体某个异常类的处理方法)
具体步骤如下:
-
编写全局异常处理类
-
定义 Spring 配置文件
五、拦截器
SpringMVC 中的 Interceptor 拦截器是非常重要和相当有用的,它的主要作用是拦截指定的用户请求,并进行相应的预处理与后处理。
- 其拦截的时间点在“处理器映射器根据用户提交的请求映射出了所要执行的处理器类,并且也找到了要执行该处理器类的处理器适配器,在处理器适配器执行处理器之前”。
- 当然,在处理器映射器映射出所要执行的处理器类时,已经将拦截器与处理器组合为了一个处理器执行链,并返回给了中央调度器。
单个拦截器的执行
自定义拦截器,需要实现 HandlerInterceptor 接口。如下:
该接口有三个方法需要实现:
➢ preHandle(request,response, Object handler):
该方法在处理器方法执行之前执行。其返回值为 boolean,若为 true,则紧接着会执行处理器方法,且会将 afterCompletion() 方法放入到一个专门的方法栈中等待执行。
➢ postHandle(request,response, Object handler,modelAndView):
该方法在处理器方法执行之后执行。处理器方法若最终未被执行,则该方法不会执行。由于该方法是在处理器方法执行完后执行,且该方法参数中包含 ModelAndView,所以该方法可以修改处理器方法的处理结果数据,且可以修改跳转方向。
➢ afterCompletion(request,response, Object handler, Exception ex):
当 preHandle()
方法返回 true 时,会将该方法放到专门的方法栈中,等到对请求进行响应的所有工作完成之后才执行该方法。即该方法是在中央调度器渲染(数据填充)了响应页面之后执行的,此时对ModelAndView 再操作也对响应无济于事。afterCompletion 最后执行的方法,清除资源,例如在Controller 方法中加入数据
拦截器中方法与处理的执行顺序如下图:
上面流程的另外一种画法如下:
举例:
-
step1:注册拦截器
<mvc:mapping/>
用于指定当前所注册的拦截器可以拦截的请求路径,而/**表示拦截所有请求 -
step2:修改 index 页面
-
Step3:修改处理器
-
Step4:修改 show 页面
-
Step5:控制台输出如下:
多个拦截器的执行
有多个拦截器时,形成拦截器链。拦截器链的执行顺序,与其注册顺序一致。需要再次强调一点的是,
- 当某一个拦截器的 preHandle()方法返回 true 并被执行到时,会向一个专门的方法栈中放入该拦截器的 afterCompletion()方法。
例子:
-
Step1: 再定义一个拦截器
在MyInterceptor1
中的PreHandle
方法返回 false;在 MyInterceptor1 中返回 true ;观察两者结果。 -
Step2: 对这几个拦截器都进行注册与执行
-
step3:执行结果如下所示:
-
第一个拦截器preHandle=true , 第二个拦截器preHandle=true
111111-拦截器的MyInterceptor的preHandle() 22222-拦截器的MyInterceptor的preHandle() =====执行MyController中的doSome方法===== 22222-拦截器的MyInterceptor的postHandle() 111111-拦截器的MyInterceptor的postHandle() 22222-拦截器的MyInterceptor的afterCompletion() 111111-拦截器的MyInterceptor的afterCompletion()
-
第一个拦截器preHandle=true , 第二个拦截器preHandle=false
111111-拦截器的MyInterceptor的preHandle() 22222-拦截器的MyInterceptor的preHandle() 111111-拦截器的MyInterceptor的afterCompletion()
- 第一个拦截器preHandle=false , 第二个拦截器preHandle=true|false
111111-拦截器的MyInterceptor的preHandle()
-
-
step4: 结果分析
在指向处理器之前,所有的方法都是按照所有方法从拦截器从上往下。在执行完拦截器方法之后,所有方法从下向上,形成一个回路,如下所示:
其中 afterCompletion 为什么也是从下往上的原因是这两个方法被压入了方法栈中,根据栈的的特性后进先出,所以 afterCompletion2 先执行。 -
从图中可以看出,只要有一个 preHandle()方法返回 false,则上部的执行链将被断开,其后续的处理器方法与 postHandle()方法将无法执行。 。因此上面的流程图也可以用下面这张图来进行表示:
过滤器与拦截器的区别
- 过滤器是servlet中的对象, 拦截器是框架中的对象
- 过滤器实现Filter接口的对象, 拦截器是实现HandlerInterceptor
- 过滤器是用来设置request,response的参数,属性的,侧重对数据过滤的。
拦截器是用来验证请求的,能截断请求。 - 过滤器是在拦截器之前先执行的。
- 过滤器是tomcat服务器创建的对象
拦截器是springmvc容器中创建的对象 - 过滤器是一个执行时间点。
拦截器有三个执行时间点 - 过滤器可以处理jsp,js,html等等
拦截器是侧重拦截对Controller的对象。 如果你的请求不能被DispatcherServlet接收, 这个请求不会行拦截器内容 - 拦截器拦截普通类方法执行,过滤器过滤servlet请求响应
拦截器的应用
需求: 只有经过登录的用户方可访问处理器,否则,将返回“无权访问”提示。
本例的登录,由一个 JSP 页面完成。即在该页面里将用户信息放入 session 中。也就是说,只要访问过该页面,就说明登录了。没访问过,则为未登录用户。(即权限检查)
-
Step1:修改 index 页面
-
Step2:定义 Controller
-
Step3: 定义 welcome 页面
-
Step4: 定义权限拦截器
在 preHandle() 方法返回 false 时,即需要使用 request 和 response 进行响应了。(因为此时不会进入处理器函数)
-
Step5: 定义 fail 页面
-
Step6:注册权限拦截器
-
Step7:定义 login 页面
-
Step8:定义 logout 页面
-
Step9:项目测试
首先在 地址栏 提交 system.do 请求
然后 再次提交 system.do 请求
访问 logout.jsp ,进行用户推出
最后 第三次提交 system.do 请求