spring boot: filter/interceptor/aop在获取request/method参数上的区别(spring boot 2.3.1)

一,filter/interceptor/aop在获取参数上有什么区别?

1,filter可以修改HttpServletRequest的参数(doFilter方法的功能),

    interceptor/aop都没有这个功能

    但它不提供到被过滤的方法的访问

    注意区分请求request的方法

2, interceptor能得到所拦截的方法名和参数名,

   不能得到参数值,

   interceptor的postHandle方法能得到ModelAndView参数,

   用来处理模板的公共参数,把页面的公共内容传递给模板

3, aspect(Aop)能得到注解所在方法的方法名和参数名、参数值,

    还可以修改方法的参数值

   另外:

   filter/interceptor都是针对请求,

   aop则是针对方法

看代码,有更直观的感受

 

说明:刘宏缔的架构森林是一个专注架构的博客,

网站:https://blog.imgtouch.com
本文: https://blog.imgtouch.com/index.php/2023/05/23/springbootfilterinterceptoraop-zai-huo-qu-requestmethod-can-shu-shang-de-qu-bie-springboot231/

         对应的源码可以访问这里获取: https://github.com/liuhongdi/

说明:作者:刘宏缔 邮箱: 371125307@qq.com

 

二,演示项目的相关信息

1,项目地址

https://github.com/liuhongdi/filterinterceptoraop

 

2,项目原理:

   我们在controller中的一个方法上,

   添加了filter/interceptor/aspect

   然后比较我们能得到哪些参数,对参数做哪些处理?

 

3,项目结构:

如图:

三,java代码说明:

1,ParamFilter.java

@Component
public class ParamFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        System.out.println("----------------filter init");
    }
   //过滤功能
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("----------------filter doFilter begin");
        //打印得到的request参数
        Enumeration paramNames = servletRequest.getParameterNames();
        while (paramNames.hasMoreElements()) {
            String paramName = (String) paramNames.nextElement();
            String[] paramValues = servletRequest.getParameterValues(paramName);
            if (paramValues.length == 1) {
                String paramValue = paramValues[0];
                if (paramValue.length() != 0) {
                    System.out.println("[filter] request parameter name:"+paramName+";value:"+paramValue);
                }
            }
        }
        //修改请求参数
        HashMap m = new HashMap(servletRequest.getParameterMap());
        m.put("newp", new String[] { "abcd" });
        m.put("v", new String[] { "filterv" });
        HttpServletRequest req = (HttpServletRequest) servletRequest;
        CustomRequestWrapper wrapRequest = new CustomRequestWrapper(req, m);
        servletRequest = wrapRequest;
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {
        System.out.println("----------------filter destroy");
    }
}

说明:filter中可以修改request参数,

但不能对方法进行访问

 

2,ValidateorInterceptor.java

@Component
public class ValidatorInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("---------------interceptor begin");

        //打印request参数
        Enumeration<?> temp = request.getParameterNames();
        if (null != temp) {
            while (temp.hasMoreElements()) {
                String en = (String) temp.nextElement();
                String value = request.getParameter(en);
                System.out.println("[interceptor] request parameters: name:"+en+";value:"+value);
            }
        }

        //打印method的相关信息
        if(handler instanceof HandlerMethod) {
            HandlerMethod handlerMethod = (HandlerMethod) handler;
            // 获取处理当前请求的 handler 信息
            System.out.println("[interceptor] method 类名:" + handlerMethod.getBeanType().getName());
            System.out.println("[interceptor] method 方法:" + handlerMethod.getMethod().getName());

            MethodParameter[] methodParameters = handlerMethod.getMethodParameters();
            for (MethodParameter methodParameter : methodParameters) {
                // 只能获取参数的名称,type,index,不能获取到参数的值
                System.out.println("[interceptor] method parameter Name: " + methodParameter.getParameterName());
                System.out.println("[interceptor] method parameter Type: " + methodParameter.getParameterType());
                System.out.println("[interceptor] method parameter Index: " + methodParameter.getParameterIndex());
            }
        }
        //sign校验无问题,放行
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    }
}

说明:interceptor中,只能得到方法的参数名,不能得到参数值

 

3,RedisRateLimiterAspect.java

@Component
@Aspect
public class RedisRateLimiterAspect {
    @Pointcut("@annotation(com.filterinterceptoraop.demo.annotation.RedisRateLimiter)")
    private void pointcut() {}

    /*
    *   around,
    *   if reach limit in time
    *   return error info
    * */
    @Around(value = "pointcut()")
    public Object requestLimit(ProceedingJoinPoint joinPoint) throws Exception {
        System.out.println("---------------aop begin");
        HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
        //打印request参数
        Enumeration<?> temp = request.getParameterNames();
        if (null != temp) {
            while (temp.hasMoreElements()) {
                String en = (String) temp.nextElement();
                String value = request.getParameter(en);
                System.out.println("[aop] request parameter name:"+en+";value:"+value);
            }
        }

        //打印方法的信息
        Object[] args = joinPoint.getArgs();
        try {
            Signature signature = joinPoint.getSignature();
            MethodSignature methodSignature = (MethodSignature)signature;
            //获取目标方法
            Method targetMethod = methodSignature.getMethod();
            String method_name = targetMethod.getName();
            System.out.println("[aop] method name:"+method_name);
            String[] paramNames = methodSignature.getParameterNames();

            Map<String, Object> nameAndArgs = new HashMap<String, Object>();
            for (int i = 0; i < paramNames.length; i++) {
                nameAndArgs.put(paramNames[i], args[i]);// paramNames即参数名
                if (paramNames[i].equals("version")) {
                    //System.out.println("version value:"+args[i]);
                    args[i] = "aopv";
                }
                System.out.println("[aop] method parameter name:"+paramNames[i]+";value:"+args[i]);
            }

            if (targetMethod.isAnnotationPresent(RedisRateLimiter.class)) {
                return joinPoint.proceed(args);
            } else {
                return joinPoint.proceed();
            }
        } catch (Throwable e) {
            e.printStackTrace();
            return null;
        }
    }
}

说明:aop中,可以得到方法的参数名和参数值,而且还可以修改方法的参数值

 

四,测试效果

1,访问url:

http://127.0.0.1:8080/home/home?v=1

然后查看控制台的输出

 

2,filter的输出

----------------filter doFilter begin
[filter] request parameter name:v;value:1

filter中我们得到了正确的request参数值

 

3,interceptor的输出

---------------interceptor begin
[interceptor] request parameters: name:v;value:filterv
[interceptor] request parameters: name:newp;value:abcd
[interceptor] method 类名:com.filterinterceptoraop.demo.controller.HomeController
[interceptor] method 方法:homeMethod
[interceptor] method parameter Name: version
[interceptor] method parameter Type: class java.lang.String
[interceptor] method parameter Index: 0
[interceptor] method parameter Name: httpServletRequest
[interceptor] method parameter Type: interface javax.servlet.http.HttpServletRequest
[interceptor] method parameter Index: 1
[interceptor] method parameter Name: httpServletResponse
[interceptor] method parameter Type: interface javax.servlet.http.HttpServletResponse
[interceptor] method parameter Index: 2
[interceptor] method parameter Name: modelMap
[interceptor] method parameter Type: class org.springframework.ui.ModelMap
[interceptor] method parameter Index: 3

请求参数v的值变成了filterv,而且增加了一个请求参数:newp,

这是filter中代码的作用

可以看到:interceptor得到了它所拦截请求对应的方法名和参数名,

但没有方法的参数值,它也不能修改request的参数和方法的参数

 

4,aop的输出:

---------------aop begin
[aop] request parameter name:v;value:filterv
[aop] request parameter name:newp;value:abcd
[aop] method name:homeMethod
[aop] method parameter name:version;value:aopv
[aop] method parameter name:httpServletRequest;value:com.filterinterceptoraop.demo.wrapper.CustomRequestWrapper@1ce466d8
[aop] method parameter name:httpServletResponse;value:org.apache.catalina.connector.ResponseFacade@45eedc13
[aop] method parameter name:modelMap;value:{}

aop中的请求参数没有变,

它可以得到方法的参数名和参数值,

在这里我们修改参数version的值为 aopv

 

5,controller中的输出:

---------------controller begin
[controller] request parameter name:v;value:filterv
[controller] request parameter name:newp;value:abcd
[controller] method parameter version:aopv

controller中接收到的方法的参数值是我们在aop中修改过的

 

五,查看spring boot版本

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v2.3.1.RELEASE)

 

posted @ 2020-07-14 13:22  刘宏缔的架构森林  阅读(2474)  评论(0编辑  收藏  举报