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

有一个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()获取请求上下文对象,并进行非空判断。如果获取到的请求上下文对象为空,我们可以根据实际情况进行异常处理或直接返回。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
答:在Mapper中获取调用的Controller方法名可以通过AOP拦截器实现。具体操作步骤如下: 1. 定义一个切面类,在该类中定义一个@Before注解的方法,用于拦截Controller层的请求。 2. 在该方法中调用Thread.currentThread().getStackTrace()方法获取方法调用栈,并通过反射获取Controller方法名。 3. 将Controller方法名添加进Mapper方法参数中,以便在Mapper中进行调用。 4. 在springboot的配置文件中配置该拦截器即可。 下面是一个简单的示例代码: ``` @Aspect @Component public class MapperAspect { @Before("execution(* com.example.demo.controller..*(..))") public void before(JoinPoint joinPoint) { String methodName = joinPoint.getSignature().getName(); String className = joinPoint.getTarget().getClass().getSimpleName(); String controllerMethodName = className + "." + methodName; MapperParam.setControllerMethodName(controllerMethodName); } } ``` ``` public class MapperParam { private static ThreadLocal<String> controllerMethodName = new ThreadLocal<>(); public static String getControllerMethodName() { return controllerMethodName.get(); } public static void setControllerMethodName(String methodName) { controllerMethodName.set(methodName); } } ``` 在Mapper接口中添加参数,并在xml文件中调用即可: ``` public interface UserMapper { User selectUserById(@Param("id") Integer id, @Param("controllerMethodName") String controllerMethodName); } ``` ``` <select id="selectUserById" resultType="com.example.demo.entity.User"> select * from user where id = #{id} and create_by = #{controllerMethodName} </select> ``` 注意,以上示例代码仅为演示目的,需要根据实际业务进行适当的修改和完善。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值