SpringBoot:拦截器的实现原理 ---- 12

拦截器的实现原理

ssss对于拦截器的实现原理,其实是非常简单的,在前几篇文章我们已经分析了,请求映射原理最核心的方法是 doDispatch(request, response),因此我们通过 deBug 方式在doDispatch处打断点学习具体的参数处理是如何工作的,其中的核心方式有获取处理器执行器,映射器,执行目标方法,页面渲染等。 在这其中,其实还穿插着一些拦截器的相关方法执行。

自定义拦截器的实现步骤

ssdsadssdas在这里插入图片描述
dsdssss我们可以发现,拦截器是一个接口,一共有三个方法,我们可以重写这三个方法,对其进行自定义拦截。

dssss①、编写HandlerInterceptor接口

@Slf4j
@Configuration
public class LoginInterceptor implements HandlerInterceptor {
    /**
     * 目标方法执行之前* @param request* @param response* @param handler* @return* @throws Exception
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        String requestURI = request.getRequestURI();
        log.info("preHandle()拦截的请求路径是()"+requestURI);
        //登陆检查逻辑
        HttpSession session = request.getSession();

        Object loginUser = session.getAttribute("loginUser");

        if(loginUser!=null){
            //放行
            return true;
        }

        //拦截住(未登录),跳转到登录页
        request.setAttribute("msg","请先登陆");
        request.getRequestDispatcher("/").forward(request,response);
        return false;
    }
    /**
     * 目标方法执行之后* @param request* @param response* @param handler* @param modelAndView* @throws Exception
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        log.info("postHandle执行{}",modelAndView);
    }
    /**
     * 页面渲染之后 * @param request* @param response* @param handler* @param ex * @throws Exception
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        log.info("afterCompletion执行异常{}",ex);
    }
}

dsss【注 1】:

dsdsssspreHandle():是执行目标方法前,执行该方法(通过方法的参数值也可以发现 -->request、response、handle)

dsdsssspostHandle():执行完成目标方法后,页面渲染前,有了返回的ModelAndView后,执行该方法。(参 -->request、response、handle、modelAndView)

dsdssssafterCompletion():页面渲染完后,执行该方法,其实主要是为了显示异常。(参:request、response、handler、ex)
dsss【注 2】:如果没有异常,三个方法按 [注 1] 中所写执行,但如果执行过程中出现异常,则改变执行顺序,具体在原理中讲。
dsdssss
dssss②、 配置拦截器

@Configuration
public class AdminWebConfig implements WebMvcConfigurer {
    /**
     * 1、编写一个拦截器实现HandlerInterceptor接口
     * 2、拦截器注册到容器中(实现WebMvcConfigurer的addInterceptors)
     * 3、指定拦截规则【如果是拦截所有,静态资源也会被拦截
     * @param registry
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LoginInterceptor())
                .addPathPatterns("/**")     //静态所有请求都被拦截,包括静态资源
                .excludePathPatterns("/","/login","/css/**","/fonts/**","/images/**","/js/**"); //放行的请求
    }
}

dssss③、拦截器执行顺序
dsdsdsdsddsss在这里插入图片描述

拦截器原理

ssss没有异常的执行流程图解:
sssadssdas在这里插入图片描述
ssss代码分析:
sdsss在这里插入图片描述
sds由上图我们可知,③,⑤,⑦是我们拦截器的三个方法。我们根据deBug依次看不同的方法逻辑:

sddss①、根据当前请求,找到HandlerExecutionChain【可以处理当前请求的handler以及handler的所有拦截器】
sddds在这里插入图片描述
sddss②、HandlerAdapter - 为当前Handler 找一个适配器

sddss③、先来 顺序执行 所有拦截器的 preHandle方法,如果其中有失败的,则逆序执行相应的afterCompletion()方法。
sdsdssssss在这里插入图片描述
sdsdssssss在这里插入图片描述
dsdsss【注】:注意逆序执行afterCompletion,有一个小细节,我们用this.interceptorIndex==i++或者–i,说明只有执行完preHandle()方法的拦截器才能逆序执行响应的afterCompletion方法。执行完后,doDispatch()方法就直接return,不会执行目标方法了。

sddss④、所有拦截器都返回True。执行目标方法。
sdsd ddss在这里插入图片描述
sddss⑤、 倒序执行所有拦截器的postHandle方法。
dsdssdds在这里插入图片描述
dsdssdd在这里插入图片描述
sddss⑥、渲染页面。

sddss⑦、页面渲染完成后也会触发所有拦截器的 afterCompletion(通过finally里的方法可以得知这一点)。
sddssdsds在这里插入图片描述
sddsdsdss并且前面的步骤有任何异常都会直接倒序触发所有拦截器的 afterCompletion。
在这里插入图片描述
dsdsss【注】:到此doDispatch()方法就结束了,拦截器的三个方法中 afterCompletion 和 preHandle 一定执行。
在这里插入图片描述

  • 6
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Spring Boot 审核功能的实现原理可以分为以下几个步骤: 1. 配置审计数据源:在 Spring Boot 中,可以通过配置文件或者代码的方式配置审计数据源,以保存所有审计事件的元数据。配置文件的示例: ``` spring.datasource.driver-class-name=com.mysql.jdbc.Driver spring.datasource.url=jdbc:mysql://localhost:3306/audit spring.datasource.username=root spring.datasource.password=123456 ``` 2. 配置审计拦截器:在 Spring Boot 中,可以通过编写审计拦截器来捕获所有请求,并在请求完成后记录审计事件的元数据。审计拦截器示例: ``` @Component public class AuditInterceptor extends HandlerInterceptorAdapter { @Autowired private AuditService auditService; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 在请求处理前记录审计事件 auditService.logEvent(request); return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { // 在请求处理完成后记录审计事件 auditService.updateEvent(request, response); } } ``` 3. 实现审计服务:在 Spring Boot 中,可以编写审计服务来将审计事件的元数据保存到审计数据源中。审计服务示例: ``` @Service public class AuditService { @Autowired private JdbcTemplate jdbcTemplate; public void logEvent(HttpServletRequest request) { // 记录请求开始时间、请求 URL 等信息 } public void updateEvent(HttpServletRequest request, HttpServletResponse response) { // 记录请求结束时间、响应状态码等信息 // 将审计事件保存到审计数据源中 jdbcTemplate.update("INSERT INTO audit_event (...) VALUES (...)"); } } ``` 通过以上步骤,就可以实现 Spring Boot 审核功能的基本原理。具体实现还需要根据业务需求进行优化和扩展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值