SpringBoot 内部方法调用,事务不起作用的原因及解决办法

2022-08-15 14:19:47 浏览数 (1)

  在做业务开发时,遇到了一个事务不起作用的问题。大概流程是这样的,方法内部的定时任务调用了一个带事务的方法,失败后事务没有回滚。查阅资料后,问题得到解决,记录下来分享给大家。

  我在这里模拟一个场景,大概的调用方式就如下面的代码这样。

  @Override

  @Transactional(rollbackFor=RuntimeException.class)

  public void insertUser(User user) {

  userMapper.insertUser(user);

  throw new RuntimeException("");

  }

  /**

  * 内部调用新增方法

  *

  * @param user

  */

  @Override

  public void invokeInsertUser(User user) {

  this.insertUser(user);

  }

  AOP使用的是动态代理的机制,它会给类生成一个代理类,事务的相关操作都在代理类上完成。内部方式使用this调用方式时,使用的是实例调用,并没有通过代理类调用方法,所以会导致事务失效。

  方法一 引入自身bean

  在类内部通过@Autowired将本身bean引入,然后通过调用自身bean,从而实现使用AOP代理操作。

  注入自身bean

  @Autowired

  @Lazy

  private UserService service;

  修改invokeInsertUser方法

  /**

  * 解决方法一 在bean中将自己注入进来

  * @param https://www.ershouyi.com/

  */

  @Override

  public void invokeInsertUser(User user) {

  this.service.insertUser(user);

  }

  方法二 通过ApplicationContext引入bean

  通过ApplicationContext获取bean,通过bean调用内部方法,就使用了bean的代理类。

  注入ApplicationContext

  @Autowired

  ApplicationContext applicationContext;

  修改invokeInsertUser方法

  /**

  * 解决方法二 通过applicationContext获取到bean

  * @param user

  */

  @Override

  public void invokeInsertUser(User user) {

  ((UserService)applicationContext.getBean("userService")).invokeInsertUser(user);

  }

  方法三 通过AopContext获取当前类的代理类

  通过AopContext获取当前类的代理类,直接通过代理类调用方法

  在引导类上添加@EnableAspectJAutoProxy(exposeProxy=true)注解

  修改invokeInsertUser方法

  /**

  * 解决方法三 通过applicationContext获取到bean

  *

  * @param user

  */

  @Override

  public void invokeInsertUser(User user) {

  ((UserService) AopContext.currentProxy()).invokeInsertUser(user);

  }

  以上就是内部方法调用时,事务不起作用的原因及解决办法。

0 人点赞