SpringBoot过滤器与拦截 器深入分析实现方法
作者:twilight0402 发布时间:2023-11-28 23:04:15
过滤器
实现过滤器需要实现 javax.servlet.Filter
接口。重写三个方法。其中 init()
方法在服务启动时执行,destroy()
在服务停止之前执行。
可用两种方式注册过滤器:
使用
FilterRegistrationBean
来注入。可使用setOrder(0)
设置过滤器的优先级,越小优先级越高。使用
@WebFilter(filterName = "myFilter2" ,urlPatterns = "/*")
配合@ServletComponentScan()
实现注入。(@Order
注解无效)
编写过滤器
package com.example.recorddemo.filters;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
public class MyFilter1 implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("初始化过滤器:" + filterConfig.getFilterName());
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
System.out.println("在请求之前做处理");
if (servletRequest instanceof HttpServletRequest) {
System.out.println(" URL:" + ((HttpServletRequest)servletRequest).getRequestURL());
}
// 调用filter链中的下一个filter
filterChain.doFilter(servletRequest, servletResponse);
System.out.println("在请求之后做处理");
}
@Override
public void destroy() {
System.out.println("销毁:MyFilter1");
}
}
注册过滤器
基于 FilterRegistrationBean
在配置类中注册一个 FilterRegistrationBean
类型的Bean。
如果没有设置
UrlPatterns
, 那么会自动关联到/*
上。如果没有设置过滤器的名字,那么会自动推理出一个过滤器名称(bean的名字)
When no URL pattern or servlets are specified the filter will be associated to ‘/*’. The filter name will be deduced if not specified.
fileter默认是enable的,将其设置为false表示关闭当前过滤器。
可通过
setOrder(0)
方法设置过滤器的优先级,如果优先级相同,则先定义的优先级更高。
@Configuration
public class FilterConfiguration {
@Bean
public FilterRegistrationBean myFilter1(){
MyFilter1 filter = new MyFilter1();
FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(filter);
// filterRegistrationBean.addUrlPatterns("/*");
// filterRegistrationBean.setEnabled(true);
return filterRegistrationBean;
}
}
基于 @WebFilter
使用
@WebFilter
修饰filter。在任意configuration类中添加
@ServletComponentScan("com.example.recorddemo.filters")
,包名可以不填。
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
@WebFilter(filterName = "myFilter2" ,urlPatterns = "/*")
public class MyFilter2 implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
// 调用filter链中的下一个filter
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {}
}
*
* 会在处理指定请求之前和之后进行相关操作,配置 * 需要两步
编写 * 类(实现
HandlerInterceptor
接口)添加已实现的 * (实现
WebMvcConfigurer
接口,并重写addInterceptors()
方法)添加
addPathPatterns()
规定拦截哪些请求。(/*
表示只拦截/
下的所有目录,但是不包括子目录,/**
表示拦截/
下的所有目录,及其子目录)
* 类:
package com.example.recorddemo.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;
/**
* @author wangchao
*/
@Component
public class MyInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
// handle可拿到执行方法的反射对象。
System.out.println("preHandle: MyInterceptor");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
// 对于RESTful 接口用处不大
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
// 可捕捉异常,但是springboot已经有了全局异常捕捉
}
}
配置 * :
package com.example.recorddemo.configuration;
import com.example.recorddemo.interceptor.MyInterceptor;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.config.annotation.InterceptorRegistration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import javax.annotation.Resource;
@Configuration
public class InterceptorConfiguration implements WebMvcConfigurer {
@Resource
MyInterceptor myInterceptor;
/**
* 添加 *
* @param registry
*/
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(myInterceptor).addPathPatterns("/**");
}
}
registry.addInterceptor()
方法会返回当前的 interceptor
, 因此可直接执行 addPathPatterns()
方法
public InterceptorRegistration addInterceptor(HandlerInterceptor interceptor) {
InterceptorRegistration registration = new InterceptorRegistration(interceptor);
this.registrations.add(registration);
return registration;
}
* 的执行顺序类似于栈,按照如下顺序执行:
preHandle-1, preHandle-2, postHandle-2, postHandle-1, afterCompletion-2, afterCompletion-1
来源:https://blog.csdn.net/twilight_karl/article/details/127641108


猜你喜欢
- 五子棋AI算法也算是一个典型的游戏AI算法,一些棋类的AI算法都可以参考实现,下面是Java实现代码棋盘抽象接口import java.ut
- 最近经朋友介绍开始玩 密传 网络游戏 升级升级,突然觉得太费键盘,于是自己用C#写了一个程序,想代替我的操作,自己去打怪物,自己升级 用这个
- 概述非对称加密算法与对称加密算法的主要差别在于非对称加密算法用于加密和解密的密钥不相同,非对称加密算法密钥分为公钥和私钥,公钥加密只能用私钥
- 1.背景选择器(位于res/drawable/,使用方法:android:background=”@drawable/XXX”) <?
- Spring中有个RestTemplate类用来发送HTTP请求很方便,本文分享一个SpringBoot发送POST请求并接收返回数据的例子
- 这篇文章主要介绍了spring cloud gateway请求跨域问题解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定
- Jetty是一个轻量级的高度可扩展的基于 java的web服务器和servlet引擎。下面是 使用 Intellij IDEA 的maven
- 前言随着敏捷开发的流行,编写单元测试已经成为业界共识。但如何来衡量单元测试的质量呢?有些管理者片面追求单元测试的数量,导致底下的开发人员投机
- 目录1、IP地址2、端口3、通信协议3.1 TCP/IP协议簇:实际上是一组协议3.2 TCP UDP对比3.3 TCP实现聊天3.4 TC
- 23种设计模式第七篇:java代理模式定义:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个
- 数组与链表的比较:数组通过下标访问的话是O(1)数组一旦声明 长度就是固定的数组的数据是物理逻辑均连续的链表增删要快一些, 数组遍历快一些长
- 本文实例讲述了Android编程设计模式之迭代器模式。分享给大家供大家参考,具体如下:一、介绍迭代器模式(Iterator Pattern)
- Java png图片修改像素rgba值import javax.imageio.ImageIO; import javax.swing.Im
- SearchView是搜索框组件,它可以让用户在文本框里输入文字,通过 * 取得用户的输入,当用户点击搜索时, * 执行实际的搜索。本文就为
- 查看公司项目代码时,存在这样一个问题:winform界面上有很多信息填写,提交
- #region 监视文件夹的变化
- 本文实例讲述了C++联合体union用法。分享给大家供大家参考。具体如下:我们应该按照C中的convention去使用union,这是我这篇
- 一、前言问题阐述:在某一场景下,我们的代码在 Service 实现相同,但却在 Controller 层访问时却希望不同的前缀可以访问。如下
- 目录目标功能点准备工作引入 Maven 依赖配置 DAO 数据层创建 JWT 工具类登录LoginFilterLoginSuccessHan
- 在不用Maven的时候,比如说以前我们用Ant构建项目,在项目目录下,往往会看到一个名为/lib的子目录,那里存放着各类第三方依赖jar文件