SpringBoot(10)AOP切入一个方法,只能获取该方法的参数

在不改变Controller的addw()方法的情况下,通过AOP切入addall()方法,可以获取到JoinPoint对象提供的信息,但无法直接获取未在Controller方法中定义的前端参数id和age。如果需要这些参数,必须修改Controller方法以接收它们。

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

如果您不想改变Controller的方法addw(),并使用AOP切入一个新的方法addall(),则addall()方法可以获取的参数仅限于在AOP切面中能够获取的参数。在AOP切面中,您可以获取方法执行时的JoinPoint对象,并从中提取方法的参数、返回值和其他相关信息。

具体来说,在这种情况下,addall()方法可以从JoinPoint对象中获取方法执行时的参数、方法名、类名、返回值等信息,但不能直接获取前端传入的参数’id’和’age’,因为它们没有被Controller方法addw()所接收。

因此,如果您需要在addall()方法中获取前端传入的参数’id’和’age’,则需要对Controller方法进行修改,以便能够接收这些参数,并在AOP切面中使用@RequestParam注解来获取它们。例如:

@Controller
public class MyController {
 
    @RequestMapping(value = "/addw", method = RequestMethod.POST)
    public void addw(@RequestParam int name, @RequestParam int id, @RequestParam int age) {
        // 在这里获取所有参数并执行一些操作
    }
 
    @RequestMapping(value = "/addall", method = RequestMethod.POST)
    public void addall() {
        // 在这里执行一些操作
    }
}

@Aspect
@Component
public class MyAspect {
 
    @AfterReturning("execution(* com.example.MyController.addw(..))")
    public void addwInterceptor(JoinPoint joinPoint, @RequestParam int name, @RequestParam int id, @RequestParam int age) {
        // 在这里可以获取请求中的所有参数,包括'id'和'age'
    }
 
    @AfterReturning("execution(* com.example.MyController.addall(..))")
    public void addallInterceptor(JoinPoint joinPoint) {
        // 在这里执行一些操作
    }
}

在这个例子中,我们将Controller方法addw()修改为接收所有参数,并在AOP切面中使用@RequestParam注解来获取它们。这样,在addw()方法执行完毕后,AOP切面就可以获取到所有参数,并在addall()方法中使用它们。同时,我们在切面类中使用@AfterReturning注解来切入addw()和addall()方法,并在这些方法执行完毕后执行一些操作。

### Spring Boot AOP 方法参数使用教程 #### 添加依赖 为了在Spring Boot项目中启用AOP功能,需引入`spring-boot-starter-aop`依赖。这使得开发者能够轻松利用面向切面编程来增强业务逻辑[^2]。 ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> ``` #### 定义切面类 创建一个带有`@Aspect`注解的Java类作为切面,在其中定义切入点表达式以及通知方法。可以访问连接点的方法签名中的参数值: ```java import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.ProceedingJoinPoint; @Aspect public class LoggingAspect { @Before("execution(* com.example.demo.controller.*.*(..)) && args(paramName)") public void logMethodArguments(String paramName){ System.out.println("Parameter value is " + paramName); } } ``` 上述代码展示了如何捕获特定包下所有控制器内的任何公共成员函数调用,并打印传入的第一个字符串类型的参数名。注意这里的`args()`用于匹配具有指定名称的实际参数列表项;如果希望获取更多参数,则可以在括号内继续添加其他形参声明[^1]。 #### 自动填充Controller层参数 对于某些场景下的通用操作(比如记录日志、设置默认值),可以通过AOP机制自动处理请求进入controller之前的工作。例如,当多个接口都需要接收相同结构的数据模型实例时,可考虑借助于环绕通知(`Around`)完成此类任务[^3]。 ```java @Around("@annotation(org.springframework.web.bind.annotation.GetMapping) && args(model,..)") public Object fillDefaultValues(ProceedingJoinPoint joinPoint, Model model)throws Throwable { // 设置默认属性... return joinPoint.proceed(); } ``` 此片段表示每当触发GET映射路径对应的处理器执行前都会先经过这段额外流程——即向视图模型对象里注入预设信息后再让原定动作按计划发生。 #### 常见问题解答 - **Q:** 如何解决因CGLIB代理造成的子类无法继承父级私有/保护级别成员变量的问题? - **A:** 可尝试调整为JDK标准接口型代理模式,或者重构源码使待共享资源变为公开静态常量等形式以便跨层次间正常传递数据[^4]。 - **Q:** 是否支持多线程环境下安全地读取修改被拦截的目标实体? - **A:** 是的,只要遵循良好的并发控制原则并合理运用同步锁机制就能保障事务一致性。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值