zoukankan      html  css  js  c++  java
  • Spring Boot实现一个监听用户请求的拦截器

    项目中需要监听用户具体的请求操作,便通过一个拦截器来监听,并继续相应的日志记录 
    项目构建与Spring Boot,Spring Boot实现一个拦截器很容易。

    Spring Boot的核心启动类继承WebMvcConfigurerAdapter

        // 增加拦截器
        @Override
        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new RequestLog());
        }
       //这个RequestLog就是我们定义的拦截器

    拦截器的编写

    public class RequestLog extends HandlerInterceptorAdapter {
    
        /**
         * 前置检查
         */
        @Override
        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
                throws Exception {
            String ip = request.getRemoteAddr();
            long startTime = System.currentTimeMillis();
            request.setAttribute("requestStartTime", startTime);
            HandlerMethod handlerMethod = (HandlerMethod) handler;
            // 获取用户token
            Method method = handlerMethod.getMethod();
            System.out.println("用户:"+ip+",访问目标:"+method.getDeclaringClass().getName() + "." + method.getName());
            return true;
        }
    
        // controller处理完成
        public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
                ModelAndView modelAndView) throws Exception {
            HandlerMethod handlerMethod = (HandlerMethod) handler;
            Method method = handlerMethod.getMethod();
            long startTime = (Long) request.getAttribute("requestStartTime");
    
            long endTime = System.currentTimeMillis();
    
            long executeTime = endTime - startTime;
    
            // log it
            if (executeTime > 1000) {
                System.out.println("[" + method.getDeclaringClass().getName() + "." + method.getName() + "] 执行耗时 : "
                        + executeTime + "ms");
            } else {
                System.out.println("[" + method.getDeclaringClass().getSimpleName() + "." + method.getName() + "] 执行耗时 : "
                        + executeTime + "ms");
            }
    
        }
    }
    
    

    我们自己实现的拦截器需要继承HandlerInterceptorAdapter,并重写如下三个方法:

    preHandle中,可以进行编码安全控制等处理; 
    在postHandle中,有机会修改ModelAndView

    还存在一个也比较重要的方法在afterCompletion中,下面介绍一些这三个方法的执行流程:

    发起请求,进入拦截器链,运行所有拦截器的preHandle方法. 
    当preHandle方法返回false时,(后面的拦截器就不再执行了)从当前拦截器往回执行所有拦截器的afterCompletion方法(不是postHandle方法哦),再退出拦截器链。 

    当preHandle方法全为true时,执行下一个拦截器,直到所有拦截器执行完。再运行被拦截的Controller。然后进入拦截器链,运行所有拦截器的postHandle方法,完后从最后一个拦截器往回执行所有拦截器的afterCompletion方法. 
    当有拦截器抛出异常时,会从当前拦截器往回执行所有拦截器的afterCompletion方法


    preHandle方法:
    返回true,映射处理器执行链将继续执行;
    当返回false时,DispatcherServlet处理器认为拦截器已经处理完了请求(这个请求已经被拦截了:http请求的status是200,但response什么也没有),而不继续执行执行链中的其它拦截器和处理器。

    http://www.itdadao.com/articles/c15a591762p0.html

  • 相关阅读:
    Support for the experimental syntax 'classProperties' isn't currently enabled
    CssSyntaxError (2:1) Unknown word 1 | > 2 | var content = require("!!./index.css");
    vue-transition实现加入购物车效果及其他动画效果实现
    Array.reduce()方法
    Object.keys()返回对象自身可枚举属性组成的数组
    vue + mixin混入对象使用
    el-table + el-form实现可编辑表格字段验证
    深圳面试题
    深圳两年面试题
    JVM中的STW和CMS
  • 原文地址:https://www.cnblogs.com/softidea/p/5987764.html
Copyright © 2011-2022 走看看