Spring 事务、异步和循环依赖有什么关系?

createh53个月前 (03-04)技术教程18

前言

在循环依赖中有一种循环依赖,就是自注入:自己依赖自己。

事务的自注入

在 Spring 自调用事务失效,你是怎么解决的? 有小伙伴提出可以自己注入自己来解决事务失效。

具体使用方式如下:

@Slf4j
@Service
public class OrderBizServiceImpl implements OrderBizService {

    // 注入自己
    @Autowired
    private OrderBizService orderBizService;

    @Override
    public void callBack() throws Exception {

        // 一系列的逻辑

        // 需要事务操作更新订单和用户金额
        orderBizService.updateOrderStatusAndUserBalance();
    }

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void updateOrderStatusAndUserBalance() throws Exception {
        // 内部是事务逻辑
    }
}

是不是发现很神奇的事情,事务生效了。

其实这里注入自己,其实是注入的一个代理对象,调事务,也是调的代理对象的事务,所以事务生效。

Spring 事务失效原因:

事务只能应用到 public 方法上才会有效;
事务需要从外部调用,Spring 自调用会失效;
建议事务注解 @Transactional 一般添加在实现类上。

异步的自注入

发现 @Transactional 注解可以自注入解决事务失效的问题,在某次开发中,自然而然想到 @Async 异步是不是也可以自注入解决循环依赖的问题。

NO, NO, NO……

事实告诉我们是不可以的!

从错误开始着手:

开始往上面反推 exposedObject == bean 是这一块出了问题。

也就是说异步的时候,再次从二级缓存中获取的和初始的不相同。

Object earlySingletonReference = getSingleton(beanName, false);

这一次获取的时候发现不同所以报错。

那就开始 Debug, 按照循环依赖的逻辑,执行到 populateBean 时,属性赋值,发现有依赖自己,此时会创建自己。

执行 singleton.getObject 方法

而此时执行 getEarlyBeanReference 先判断 InfrastructureAdvisorAutoProxyCreator true 调用 wrapIfNecessary 判断是否生成一个代理对象,这里并没有生成代理对象。

然后开始执行异步的 AsyncAnnotationBeanPostProcessor 判断为 false。所以没有执行异步的生成代理对象逻辑。

那就继续往下看

进入到 initializeBean 的逻辑,有一部分叫做 applyBeanPostProcessorsAfterInitialization

方面小伙伴搜索,所以贴出来代码关键字。IDEA 使用 ? + Shift + F 搜索。

循环执行后置处理器:

发现执行完 AsyncAnnotationBeanPostProcessor 这个 PostProcessor 后,对象被改变了。从而导致二级缓存和当前的 Bean 不同。

以上也就是为什么 @Async 自调用不可以,因为在后面初始化阶段被代理修改了对象。

@Transactional 为什么可以呢?

先判断 InfrastructureAdvisorAutoProxyCreator true 生成一个代理对象。

事务的处理器 PersistenceExceptionTranslationPostProcessor 也没有执行。

继续 Debug 关注 applyBeanPostProcessorsAfterInitialization

执行结束,发现 Bean 没有发生改变。

总结

  • @Transactional: 是在循环依赖从二级缓存升到三级缓存的时候已经生成了代理对象。
  • @Async: 是在初始化阶段(initializeBean)去生成代理对象。然后 @Async 导致后面判断 exposedObject == bean 为 false ,从而抛出异常。

可以看出图中有两处会执行 BeanPostProcessor :

  1. 在 singletonFactory.getObject 时,如果是 SmartInstantiationAwareBeanPostProcessor 的子类会执行 getEarlyBeanReference 方法。
  2. 在 initializeBean 的 applyBeanPostProcessorsAfterInitialization 时会执行所有 BeanPostProcessor 的 postProcessAfterInitialization 的方法。

也有其他的地方在执行后置处理器,比如
applyBeanPostProcessorsBeforeInitialization ,只不过这里关注这
俩处

而这两处都有可能生成代理对象, @Transactional 是在 getEarlyBeanReference 处生成的代理对象,所以后面判断 Bean 是否被改变时为 true,而 @Async 是在后面异步生成了代理对象,所以判断不通过。

至此,分析完毕,错误之处,欢迎指正。

作者:刘志航

出处:https://www.cnblogs.com/liuzhihang/p/spring-trans-async.html

相关文章

Java项目中启动报错循环依赖问题解决

项目启动时候失败,有时候经常看到这样的错误信息:Caused by: org.springframework.beans.factory.BeanCurrentlyInCreationExceptio...

为什么在Spring中有时候不应该使用@Autowired进行依赖注入?

1. 前言??注解@Autowired,相信对于我们Java开发者而言并不陌生吧,在SpringBoot或SpringCloud框架中使用那是非常的广泛。但是当我们使用IDEA编辑器开发代码的时候,经...

从@Resource到@Autowired:依赖注入中的两大战将,该选择哪一个

Spring Framework是一款广泛应用于Java开发的轻量级框架。Spring Framework提供了很多核心功能,如依赖注入(Dependency Injection)和面向切面编程(As...

几个可视化工具

几个可视化工具可视化 Docker Compose 文件下一个工具专门针对 Docker 用户,但它仍然与语言无关。要可视化docker-compose.yml,您可以使用docker-compose...