一、二者区别
参考网址:
https://www.cnblogs.com/protected/p/6649587.html
https://www.cnblogs.com/junzi2099/p/8022058.html
Fliter和Interceptor二者主要区别见下表
- Filter是基于函数回调(doFilter()方法)的,而Interceptor则是基于Java反射的(AOP思想)。
- Filter依赖于Servlet容器,而Interceptor不依赖于Servlet容器。
- Filter对几乎所有的请求起作用,而Interceptor只能对action请求起作用。
- Interceptor可以访问Action的上下文,值栈里的对象,而Filter不能。
- 在action的生命周期里,Interceptor可以被多次调用,而Filter只能在容器初始化时调用一次。
Filter和Interceptor的执行顺序
过滤前-拦截前-action执行-拦截后-过滤后
二、二者的实现方式
1. Filter
Filter主要采用实现Filter接口的形式完成初始化
//执行顺序,值越小,越先执行
@Order(1)
//指明这是一个Filter
@WebFilter(filterName = "testFilter1", urlPatterns = "/*")
public class YkFilterDemo implements Filter {
/**
* 当容器初始化 Filter 时调用,该方法在 Filter 的生命周期只会被调用一次,一般在该方法中初始化一些资源,
* FilterConfig 是容器提供给 Filter 的初始化参数,在该方法中可以抛出 ServletException。
* init 方法必须执行成功,否则 Filter 可能不起作用,出现以下两种情况时,web 容器中 Filter 可能无效:
* 1)抛出 ServletException 2)超过 web 容器定义的执行时间。
* @param filterConfig
* @throws ServletException
*/
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
/**
* Web 容器每一次请求都会调用该方法。
* 该方法将容器的请求和响应作为参数传递进来, FilterChain 用来调用下一个 Filter
* @param request
* @param response
* @param chain
* @throws IOException
* @throws ServletException
*/
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
//do something here
//链式调用
chain.doFilter(request, response);
}
/**
* 当容器销毁 Filter 实例时调用该方法,可以在方法中销毁资源,该方法在 Filter 的生命周期只会被调用一次。
*/
@Override
public void destroy() {
}
}
2. Interceptor
先编写Interceptor类
public class YkInterceptorDemo implements HandlerInterceptor {
/**
* 在handler执行之前,返回 boolean 值,true 表示继续执行,false 为停止执行并返回。
* @param request
* @param response
* @param handler
* @return
* @throws Exception
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
return true;
}
/**
* 在handler执行之后, 可以在返回之前对返回的结果进行修改
* @param request
* @param response
* @param handler
* @param modelAndView
* @throws Exception
*/
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception {
}
/**
* 在请求完全结束后调用,可以用来统计请求耗时等等
* @param request
* @param response
* @param handler
* @param ex
* @throws Exception
*/
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception {
}
}
然后需要将我们编写好的类注册到SpringMvc中即可
@Configuration
public class YkMvcConfig extends WebMvcConfigurationSupport {
@Override
protected void addInterceptors(InterceptorRegistry registry) {
super.addInterceptors(registry);
registry.addInterceptor(new YkInterceptorDemo()).addPathPatterns("/**");
}
}
三、二者如何使用@Autowired
二者如果使用方法不对,直接@Autowired很容易爆出空指针异常,下面教大家如何处理:
Filter @Autowired报错
Filter本来就是依赖于servlet容器的,所以对于spring容器里面的东西是不能直接@Autowired出来的。于是要么:
1.强制注册进入spinrg容器里面
@Override
public void init(FilterConfig filterConfig) throws ServletException {
SpringBeanAutowiringSupport.processInjectionBasedOnServletContext(this, filterConfig.getServletContext());
}
2.从webApplication中获取
ServletContext sc = ((HttpServletRequest) request).getSession().getServletContext();
WebApplicationContext webApplicationContext = WebApplicationContextUtils.getWebApplicationContext(sc);
webApplicationContext.getEnvironment().getProperty("server.servlet.context-path");
webApplicationContext.getBean("redisVarPool");
Inteceptor @Autowired报错
本来Inteceptor就是是springMvc里面的东西,理应使用@Autowried没问题,但是就是因为没有放到bean里面,直接简单的new YkInterceptorDemo()出来这样子就有问题。改写如下:
这样子在YkInterceptorDemo里面的@Autowired对象就不会为空了。
四、自我总结
- Filter是一种过滤器,过滤器的本质是为了漏掉东西,所以比如做一些请求参数头的安全性过滤,低俗文字,危险字符的过滤。
- Inteceptor是一种拦截器,对于某些controller请求进行拦截。很明显的AOP切面思想。
- 在技术实现的时候,既可以用Filter,也可以用Inteceptor,主要是这两个是不同公司的产品。相比之下,Filter会更加底层一些,如果要用到spring的IOC管理,那最好还是用Inteceptor,那会简便很多。