SpringBoot(11)AOP切入方法在不改变Controller任何代码的情况下获取额外参数

在不改变Controller代码的情况下,通过AOP拦截器使用HttpServletRequest获取前端传入的id和age参数,实现方法间的参数传递。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

有一个Controller的方法addw(int name){},返回值为void,映射路径是/addw,前端传入的参数有name、id、age,现在希望在不改变Controller的方法的前提下,使用AOP切入一个新的方法addall(),此时,addall方法可以获取的参数为name,有没有什么办法,在不改变Controller任何代码的情况下获取到id、age

在不改变Controller代码的情况下,addall方法无法直接获取前端传入的id、age参数。因为这些参数是通过HTTP请求的query参数或body参数传递的,而AOP只能拦截Controller方法的调用,无法直接拦截HTTP请求的处理过程。

但是可以通过一些技巧来实现在不改变Controller任何代码的情况下获取到id、age参数,例如:

  • 使用ThreadLocal保存参数:在Controller方法中将参数保存到ThreadLocal中,然后在AOP拦截的方法中获取ThreadLocal中的参数。这种方法需要在Controller方法中加入一些额外的代码。

  • 使用HttpServletRequest获取参数:在AOP拦截的方法中,通过HttpServletRequest对象获取请求中的参数。这种方法不需要在Controller方法中加入额外的代码,但需要在AOP拦截的方法中添加HttpServletRequest对象参数,并且需要注意参数名与请求参数名的一致性。

使用HttpServletRequest获取参数的解决方案进行举例

下面是使用HttpServletRequest获取参数的示例代码:

Controller方法:

@RequestMapping(value = "/addw", method = RequestMethod.POST)
public void addw(@RequestParam("name") String name) {
    // Controller方法中只获取了name参数
    System.out.println("Controller方法中获取的参数 name:" + name);
}

AOP拦截方法:

@Around("execution(* com.example.controller.TestController.addw(..))")
public void addall(ProceedingJoinPoint joinPoint) throws Throwable {
    // 获取HttpServletRequest对象
    HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
    // 通过HttpServletRequest对象获取参数
    String id = request.getParameter("id");
    String age = request.getParameter("age");
    System.out.println("AOP拦截方法中获取的参数 id:" + id);
    System.out.println("AOP拦截方法中获取的参数 age:" + age);
    // 调用Controller方法
    joinPoint.proceed();
}

在上述代码中,AOP拦截方法中通过RequestContextHolder获取当前请求的HttpServletRequest对象,然后通过调用HttpServletRequest的getParameter方法获取请求中的参数。注意需要在AOP拦截方法中添加HttpServletRequest对象参数,并且需要与请求参数名一致。同时需要注意,如果请求参数名与参数类型不一致,则会出现参数值为null的情况。

RequestContextHolder.getRequestAttributes()没有获取到数据

如果RequestContextHolder.getRequestAttributes()没有获取到数据,可能是因为在AOP拦截方法执行的过程中,当前线程的请求对象已经被清除,导致无法获取请求信息。这种情况通常出现在异步线程池等情况下。

为了解决这个问题,可以尝试使用RequestContextHolder.getRequestAttributes()获取请求上下文对象,并在拦截方法中进行非空判断。

下面是修改后的示例代码:

@Around("execution(* com.example.controller.TestController.addw(..))")
public void addall(ProceedingJoinPoint joinPoint) throws Throwable {
    // 获取HttpServletRequest对象
    ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
    if (requestAttributes == null) {
        // 请求上下文对象为空,处理异常或者直接返回
        return;
    }
    HttpServletRequest request = requestAttributes.getRequest();
    // 通过HttpServletRequest对象获取参数
    String id = request.getParameter("id");
    String age = request.getParameter("age");
    System.out.println("AOP拦截方法中获取的参数 id:" + id);
    System.out.println("AOP拦截方法中获取的参数 age:" + age);
    // 调用Controller方法
    joinPoint.proceed();
}

在上述代码中,我们使用RequestContextHolder.getRequestAttributes()获取请求上下文对象,并进行非空判断。如果获取到的请求上下文对象为空,我们可以根据实际情况进行异常处理或直接返回。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值