Spring获取当前类的代理类(用于类内部方法调用事务不生效的情况)

本文探讨了在Spring中如何通过@Autowired、@Resource注解或BeanFactory来获取并使用事务代理对象,强调了@EnableAspectJAutoProxy(exposeProxy=true)的重要性。文章解释了为何需要设置exposeProxy为true,因为代理对象通常存储在ThreadLocal中,并展示了获取代理对象的代码片段。同时,提到了在使用Spring声明式事务时应注意事务的传播特性,以避免潜在问题。

1. 使用@Autowired注解或者@Resource注解注入自己,然后使用注入的代理类去调用事务方法。如下:

2. 注入BeanFactory容器,使用容器获取代理类,然后使用代理类调用事务方法。如下:

3. 在类上面添加注解@EnableAspectJAutoProxy(exposeProxy = true),然后在类中获取当前类的代理类,使用代理类调用事务方法。如下:

注:如果不添加注解的话,可能会出现以下错误

java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.  

解析:为什么需要exposeProxy = true

  1. 代理对象的获取其实是从线程的ThreadLocal中获取的,如下:
  2. 对象是什么时候放进去的呢,查看JdkDynamicAopProxy和CglibAopProxy,其中有以下代码,所以注解中需要先置为true才会将代理对象设置进去。

备注:使用Spring的声明式事务时需要注意事务的传播特性,避免使用时出现问题。

### 原理 Spring采用动态代理机制来实现事务控制。当一个方法被`@Transactional`注解标记时,Spring会为该生成一个代理类。在外部调用这个被注解的事务方法时,实际执行的是代理类里的方法代理类会在方法执行前后进行事务的开启、提交或回滚等操作。 但在同一中,事务方法调用事务方法时,由于是在内部进行调用调用的是原始对象的方法,而不是代理类方法。原始对象在调用方法时,不会再触发代理,也就不会开启新的事务,非事务方法会在当前事务方法事务上下文里执行 [^2]。 ### 影响 - **事务一致性问题**:非事务方法可能会对数据库进行修改操作,如果在执行过程中出现异常,由于没有独立的事务控制,这些修改可能无法回滚,从而破坏了数据的一致性。 - **事务传播行为失效**:即使非事务方法上配置了事务传播行为,由于没有触发代理,事务传播行为也不会生效,无法按照预期的方式管理事务。 ### 处理方式 - **自己注入自己**:在中注入自身的实例,通过注入的实例来调用事务方法,这样调用的就是代理类方法,能触发事务代理。示例代码如下: ```java import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; @Service public class MyService { @Autowired private MyService self; @Transactional public void transactionalMethod() { // 调用事务方法 self.nonTransactionalMethod(); } public void nonTransactionalMethod() { // 非事务方法的业务逻辑 } } ``` - **获取当前代理对象**:使用`AopContext.currentProxy()`获取当前代理对象,通过代理对象调用事务方法。需要在配置上添加`@EnableAspectJAutoProxy(exposeProxy = true)`来暴露代理对象。示例代码如下: ```java import org.springframework.aop.framework.AopContext; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; @Service @org.springframework.context.annotation.EnableAspectJAutoProxy(exposeProxy = true) public class MyService { @Transactional public void transactionalMethod() { MyService proxy = (MyService) AopContext.currentProxy(); proxy.nonTransactionalMethod(); } public void nonTransactionalMethod() { // 非事务方法的业务逻辑 } } ``` - **获取当前的bean来进行调用**:使用`SpringUtil`工具获取当前的bean实例,通过该实例调用事务方法。可以使用Hutool的`SpringUtil`或自己封装一个`SpringUtil`。示例代码如下: ```java import com.hutool.extra.spring.SpringUtil; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; @Service public class MyService { @Transactional public void transactionalMethod() { MyService myService = SpringUtil.getBean(MyService.class); myService.nonTransactionalMethod(); } public void nonTransactionalMethod() { // 非事务方法的业务逻辑 } } ``` - **反射调用**:通过反射机制调用事务方法,解决事务失效问题。示例代码如下: ```java import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.lang.reflect.Method; @Service public class MyService { @Transactional public void transactionalMethod() throws Exception { Method method = this.getClass().getMethod("nonTransactionalMethod"); method.invoke(this); } public void nonTransactionalMethod() { // 非事务方法的业务逻辑 } } ```
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值