
惊呆了,Servlet Filter和Spring MVC Interceptor的实现居然这么简单
介绍
在工作中,我们经常要和Servlet Filter,Spring MVC Interceptor打交道,虽然我配置写的很6,但是对实现方式还是挺好奇的,于是看了一下源码,用Demo的方式来分析一下这两者是怎么工作的。
Servlet Filter
Filter的使用
可能很多小伙伴没怎么用过Filter,我就简单演示一下
1.在web*.xml中配置2个Filter*
2.实现如下,略去了init方法和destroy方法
3.然后你访问任意一个servlet方法,LogFilter和ImageFilter的doFilter方法都会执行
如果你在一个Filter方法后不加**chain.doFilter(request, response)**则后续的Filter和Servlet都不会执行,这是为什么呢?看完我手写的Demo你一下就明白了
可以看到Filter可以在请求到达Servlet之前做处理,如
- 请求编码
- 敏感词过滤等
有兴趣的小伙伴可以看看相关的源码
手写Filter的实现
Servlet接口,任何一个web请求都会调用service方法
拦截器接口
拦截器链对象
测试例子
如果任意一个Filter方法的最后不加上chain.doFilter(),则后面的拦截器及Servlet都不会执行了。相信你看完ApplicationFilterChain类的doFilter方法一下就明白了,就是一个简单的递归调用
Spring MVC Interceptor
Interceptor的使用
以前写过一篇拦截器应用的文章,有想了解使用方式的小伙伴可以看一下
用Spring MVC拦截器做好web应用的安保措施
今天就来分析一下拦截器是怎么实现的?可以通过以下方式实现拦截器
- 实现HandlerInterceptor接口
- 继承HandlerInterceptorAdapter抽象类,按需重写部分实现即可,(HandlerInterceptorAdapter也实现了HandlerInterceptor接口)
总而言之拦截器必须必须实现了HandlerInterceptor接口
HandlerInterceptor有如下3个方法
boolean preHandler():在controller执行之前调用void postHandler():controller执行之后,且页面渲染之前调用void afterCompletion():页面渲染之后调用,一般用于资源清理操作
这个图应该很好的显示了一个请求可以被拦截的地方
- Servlet Filter是对一个请求到达Servlet的过程进行拦截
- 而HandlerInterceptor是当请求到达DispatcherServlet后,在Controller的方法执行前后进行拦截
手写Interceptor的实现
我来手写一个Demo,你一下就能明白了
拦截接口,为了方便我这里就只定义了一个方法
定义如下2个拦截器
存放拦截器的容器
演示DispatcherServlet的调用过程
如果任意一个Interceptor返回false,则后续的Interceptor和Controller中的方法都不会执行原因在Demo中显而易见
当想对请求增加新的过滤逻辑时,只需要定义一个拦截器即可,完全符合开闭原则。
不知道你意识到没有Servlet Filter和Spring MVC Interceptor都是用责任链模式实现的
来看看DispatcherServlet是怎么做的?和我们上面写的demo一模一样
我们用servlet写web应用时,一个请求地址写一个Servlet类。而用了spring mvc后,整个应用程序只有一个Servlet即DispatcherServlet,所有的请求都发送到DispatcherServlet,然后通过方法调用的方式执行controller的方法
DispatcherServlet的doDispatch方法源码如下,省略了一部分逻辑(所有的请求都会执行这个方法)
Interceptor可以有如下用处
- 记录接口响应时间
- 判断用户是否登陆
- 权限校验等
可以看到Servlet Filter和Spring MVC Interceptor都能对请求进行拦截,只不过时机不同。并且Servlet Filter是Servlet的规范,而Spring MVC Interceptor只能在Spring MVC中使用
文章转载自公众号:Java识堂
