AopContext.currentProxy遇到的坑

本文探讨了Spring AOP中service层自调用导致的日志遗漏问题,并介绍了如何使用AopContext.currentProxy()解决该问题。同时指出使用此方法时的一些限制条件,如被调用方法必须为public修饰。

https://www.cnblogs.com/aligege/p/13632855.html

Spring的切面功能,是通过代理的方式来实现的,实现方式有JDK动态代理方式及Cglib的方式。

有时需要在service层通过AOP做一些日志、权限、监控等功能,但在service中进行自调用时,无法再次走进代理类中,因此会导致漏日志的情况出现。

此时,可通过AopContext.currentProxy()来解决问题。用AopContext.currentProxy().xxxx()的方式代替this.xxxx()的自调用方式。
AopContext.currentProxy()的本质是使用的ThreadLocal生成本地代理,这样的做法可能影响性能,后续文章对ThreadLocal的内部原理和性能进行进一步深入!
AopContext.currentProxy()的局限性:

  1)AopContext.currentProxy()调用的方法中,如果此方法再调用其他的service,则此方法必须是public的修饰符。否则会发现通过@Autowired注入的service值为null。

 

 

事情是这样的,为了使用事务,所以用到了AopContext.currentProxy()方法。

mainMethiod{

methodOne

methodTwo

}

methodOne正常执行,methodTwo方法里一个mapper报空指针。通过debug发现,执行到methodTwo后,service换了一个bean(id不同了)。

百度了一翻后发现上文,检查代码,果然methodTwo是private的。改为public后正常。

 

### AopContext.currentProxy的作用及使用示例 #### 作用 `AopContext.currentProxy()` 是 Spring AOP 提供的一个工具方法,其主要作用是获取当前代理对象的实例。在 Spring AOP 中,当一个类的方法被切面逻辑(如日志记录、事务管理等)增强后,Spring 会为该类创建一个代理对象。通过 `AopContext.currentProxy()`,可以访问到这个代理对象,从而确保在调用其他方法时,切面逻辑能够被正确触发。 在某些场景下,例如类内部的自调用(即一个方法调用另一个方法),SpringAOP 机制可能无法正确应用切面逻辑。这是因为 Spring 默认通过代理对象来管理切面逻辑的执行,而直接通过 `this` 调用的方法并不会经过代理对象。此时,通过 `AopContext.currentProxy()` 获取代理对象并调用目标方法,可以确保切面逻辑被正确执行 [^1]。 此外,`AopContext.currentProxy()` 的设计还体现了 Spring AOP 的一些重要架构理念,如上下文隔离、线程安全性和透明代理机制。它确保了不同线程之间的代理对象不会相互干扰,并且能够动态地返回当前线程的代理实例 [^2]。 #### 使用示例 以下是一个使用 `AopContext.currentProxy()` 的示例,展示了如何通过代理对象调用方法以确保切面逻辑的执行。 ```java import org.springframework.aop.framework.AopContext; import org.springframework.stereotype.Service; @Service public class UserServiceImpl implements UserService { @Override public void performAction() { // 直接调用,可能不会触发切面逻辑 performAnotherAction(); // 通过 AopContext 获取代理对象并调用方法,确保切面逻辑生效 ((UserServiceImpl) AopContext.currentProxy()).performAnotherAction(); } @Override public void performAnotherAction() { // 方法逻辑 System.out.println("Performing another action."); } } ``` 在这个示例中,`performAction()` 方法尝试通过 `this` 调用 `performAnotherAction()`,但由于 Spring AOP 的机制,这种方式可能不会触发切面逻辑。为了确保切面逻辑生效,代码通过 `AopContext.currentProxy()` 获取代理对象,并通过代理对象调用 `performAnotherAction()`,从而保证切面逻辑被正确执行 [^3]。 另一个示例是直接通过 `AopContext.currentProxy()` 获取代理对象并进行调用: ```java import org.springframework.aop.framework.AopContext; import org.springframework.stereotype.Service; @Service public class MessageSendConfigServiceImpl { public void findSendConfigUniqueWithCache(Query query) { // 获取代理对象并调用方法 ((MessageSendConfigServiceImpl) AopContext.currentProxy()).findSendConfigUniqueWithCache(query); } } ``` 在这个示例中,通过 `AopContext.currentProxy()` 获取代理对象,并调用目标方法 `findSendConfigUniqueWithCache()`,从而确保切面逻辑被正确触发 [^4]。 #### 注意事项 - 使用 `AopContext.currentProxy()` 时,必须确保当前线程中存在代理对象。如果在没有代理对象的情况下调用该方法,可能会抛出异常。 - `AopContext.currentProxy()` 返回的是当前线程的代理对象,因此在多线程环境中,每个线程都会有自己的代理对象实例,避免了线程安全问题。 - 在使用 `AopContext.currentProxy()` 时,需要显式地将返回值转换为目标类型的代理对象,以便调用相关方法。 通过合理使用 `AopContext.currentProxy()`,可以有效解决 Spring AOP 中的自调用失效问题,并确保切面逻辑的正确执行。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值