@Transactional失效——同一个类中方法调用导致,解决方案!

@transaction注解失效的6种场景:https://blog.csdn.net/zhangkaixuan456/article/details/108882496

 同一个类中方法调用,导致@Transactional失效

开发中避免不了会对同一个类里面的方法调用,比如有一个类Test,它的一个方法A,A再调用本类的方法B(不论方法B是用public还是private修饰),但方法A没有声明注解事务,而B方法有。则外部调用方法A之后,方法B的事务是不会起作用的。这也是经常犯错误的一个地方。


    @GetMapping("/test")
    private Integer A() throws Exception {
        CityInfoDict cityInfoDict = new CityInfoDict();
        cityInfoDict.setCityName("2");
        /**
         * B 插入字段为 3的数据
         */
        int insert=insertB();
        /**
         * A 插入字段为 2的数据
         */
        int insert = cityInfoDictMapper.insert(cityInfoDict);
 
        return insert;
    }
 
    @Transactional()
    public Integer insertB() throws Exception {
        CityInfoDict cityInfoDict = new CityInfoDict();
        cityInfoDict.setCityName("3");
        cityInfoDict.setParentCityId(3);
 
        return cityInfoDictMapper.insert(cityInfoDict);
    }

  注解失效原因分析:

  其实这还是由于使用Spring AOP代理造成的,因为只有当事务方法被当前类以外的代码调用时,才会由Spring生成的代理对象来管理。

  解决方案:

  利用spring提供的动态代理机制。

1.引入spring动态代理:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
    <version>2.3.3.RELEASE</version>
</dependency>

2.方法调用改成动态代理调用:


    @GetMapping("/test")
    private Integer A() throws Exception {
        CityInfoDict cityInfoDict = new CityInfoDict();
        cityInfoDict.setCityName("2");
       
       // B 插入字段为 3的数据
        Test test = (Test)AopContext.currentProxy();

        int insert=test.insertB();
       
        int insert = cityInfoDictMapper.insert(cityInfoDict);
 
        return insert;
    }
 
    @Transactional()
    public Integer insertB() throws Exception {
        CityInfoDict cityInfoDict = new CityInfoDict();
        cityInfoDict.setCityName("3");
        cityInfoDict.setParentCityId(3);
 
        return cityInfoDictMapper.insert(cityInfoDict);
    }

可能会提示:Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.

在代理类上添加:

@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true) 

问题解决!!!!!!

  • 18
    点赞
  • 51
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
在Java@Transactional注解是Spring框架用来控制事务的注解。当我们在一个类调用事务方法时,如果这个类自身被注入到Spring容器,那么@Transactional注解是可以正常工作的。但是,如果这个类是通过new关键字创建的对象,而不是通过Spring容器注入的对象,那么@Transactional注解就会失效。 这是因为@Transactional注解的实现原理是通过Spring AOP动态代理技术生成一个代理类来处理事务。当一个类被注入到Spring容器时,Spring会对这个类进行增强,生成一个代理类,从而使@Transactional注解生效。但是如果我们通过new关键字创建一个对象,这个对象并没有被Spring容器管理,也就没有被增强,因此@Transactional注解就无法生效了。 举个例子,假设我们有一个UserService类,其有一个addUser方法使用了@Transactional注解来控制事务: ```java @Service public class UserService { @Autowired private UserDao userDao; @Transactional public void addUser(User user) { userDao.addUser(user); } } ``` 如果我们在另一个类创建一个UserService对象,并调用其addUser方法,那么@Transactional注解就会失效: ```java public class OtherClass { public void doSomething() { UserService userService = new UserService(); // 创建UserService对象 User user = new User(); user.setName("test"); userService.addUser(user); // 调用addUser方法@Transactional注解失效 } } ``` 因此,在使用@Transactional注解时,我们需要注意在哪些地方创建了对象,确保事务注解能够正常工作。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhangkaixuan456

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值