过滤器英文叫 Filter,是 JavaEE 的标准,依赖于 Servlet 容器,可以按照指定顺序配置多个。常用来配置请求编码以及过滤一些非法参数,垃圾信息或者是网站登录验证码。
目前主要有三种实现方式
主要分为两步
public class JavaEncodeFilter implements Filter {
private String encoding = "UTF-8";
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
servletRequest.setCharacterEncoding(encoding);
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
<!-- filter -->
<filter>
<filter-name>javaEncodeFilter</filter-name>
<filter-class>edu.jiahui.convertlabgateway.service.filter.JavaEncodeFilter</filter-class>
<init-param>
<param-name>forceEncoding</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>javaEncodeFilter</filter-name>
<url-pattern>/*
主要分为三步
@SpringBootApplication
@ServletComponentScan(basePackages = "edu.jiahui.convertlabgateway")
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);;
}
}
@Order(1)
@WebFilter(filterName = "sprintBootEncodeFilter", urlPatterns = {"/*"})
public class SprintBootEncodeFilter implements Filter {
private String encoding = "UTF-8";
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
servletRequest.setCharacterEncoding(encoding);
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
WebFilter由Servlet3.0提供,用来代替在web.xml文件中的配置,WebFilter 的常用属性介绍:
标签。
的取值。Spring的OncePerRequestFilter类实际上是一个实现了Filter接口的抽象类。spring对Filter进行了一些封装处理。 顾名思义,它能够确保在一次请求中只通过一次filter,而需要重复的执行,主要是为了解决servlet不同版本的差异:
@Order(5)
@WebFilter(filterName = "sprintEncodeFilter", urlPatterns = {"/*"})
public class SprintEncodeFilter extends OncePerRequestFilter {
private String encoding = "UTF-8";
@Override
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
request.setCharacterEncoding(encoding);
filterChain.doFilter(request, response);
}
}
拦截器 Interceptor 不依赖 Servlet 容器,依赖 Spring 等 Web 框架,在 SpringMVC 框架中是配置在SpringMVC 的配置文件中,在 SpringBoot 项目中也可以采用注解的形式实现。
拦截器是 AOP 的一种应用,底层采用 Java 的反射机制来实现的。与过滤器一个很大的区别是在拦截器中可以注入 Spring 的 Bean,能够获取到各种需要的 Service 来处理业务逻辑,而过滤器则不行。
主要分为两步
1.自定义类继承org.springframework.web.servlet.handler.HandlerInterceptorAdapter
@Service
public class TestInterceptor extends HandlerInterceptorAdapter {
private String encoding = "UTF-8";
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
request.setCharacterEncoding(encoding);
return true;
}
}
2.添加拦截器的配置信息
@Configuration
public class InterceptorConfig implements WebMvcConfigurer {
/**
* 不需要登录拦截的url
*/
final String[] notLoginInterceptPaths = {
"/home",
"rpc/**"
};
@Resource
private TestInterceptor testInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(testInterceptor)
.addPathPatterns("/**")
.excludePathPatterns(notLoginInterceptPaths);
// 可以添加多个拦截器的配置
// registry.addInterceptor(pageInterceptor)
// .addPathPatterns("/**");
}
}
1.拦截器是基于java的反射机制的,而过滤器是基于函数回调。
2.拦截器不依赖与servlet容器,过滤器依赖与servlet容器。
3.拦截器只能对Controller请求起作用,而过滤器则可以对几乎所有的请求起作用。
4.拦截器可以访问Controller上下文、值栈里的对象,而过滤器不能访问。
5.在Controller的生命周期中,拦截器可以多次被调用,而过滤器只能在容器初始化时被调用一次。
6.拦截器可以获取IOC容器中的各个bean,而过滤器就不行,这点很重要,在拦截器里注入一个service,可以调用业务逻辑。
层次图解:
执行顺序图解:
小结:Filter是在Servlet规范规定的,只能用于Web程序中,只在Servlet前后起作用。而拦截器是基于Spring的AOP,能够深入到方法前后,异常抛出前后等,因此拦截器的使用具有更大的弹性,在Spring框架的程序中,优先考虑使用拦截器。