从源码层面剖析@Transactional实现原理

1.组件注入

在SpringBoot中,通过@EnableTransactionManagement注解开启事务管理,而该注解上添加了@Import(TransactionManagementConfigurationSelector.class)注解,通过TransactionManagementConfigurationSelector向容器中注入了两个组件AutoProxyRegistrarProxyTransactionManagementConfiguration

其中AutoProxyRegistrar实现了ImportBeanDefinitionRegistrar接口,给容器中添加了InfrastructureAdvisorAutoProxyCreator后置处理器

ProxyTransactionManagementConfiguration给容器中添加了BeanFactoryTransactionAttributeSourceAdvisor事务增强器,并未增强器设置了TransactionAttributeSource和事务拦截器TransactionInterceptor

@Configuration
public class ProxyTransactionManagementConfiguration extends AbstractTransactionManagementConfiguration {

    @Bean(name = TransactionManagementConfigUtils.TRANSACTION_ADVISOR_BEAN_NAME)
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public BeanFactoryTransactionAttributeSourceAdvisor transactionAdvisor() {
        BeanFactoryTransactionAttributeSourceAdvisor advisor = new BeanFactoryTransactionAttributeSourceAdvisor();
        advisor.setTransactionAttributeSource(transactionAttributeSource());
        advisor.setAdvice(transactionInterceptor());
        advisor.setOrder(this.enableTx.<Integer>getNumber("order"));
        return advisor;
    }

    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public TransactionAttributeSource transactionAttributeSource() {
        return new AnnotationTransactionAttributeSource();
    }

    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public TransactionInterceptor transactionInterceptor() {
        TransactionInterceptor interceptor = new TransactionInterceptor();
        interceptor.setTransactionAttributeSource(transactionAttributeSource());
        if (this.txManager != null) {
            interceptor.setTransactionManager(this.txManager);
        }
        return interceptor;
    }
}
2.代理对象的生成

前面在分析AOP原理时讲到,在创建代理对象前,需要先获取该对象的拦截器链,包括:1. 获取所有Advisor 2. 在所有Advisor中挑选适用的Advisor

BeanFactoryTransactionAttributeSourceAdvisor同样实现了 Advisor 接口。那么在获取所有增强器时自然会将此 bean 提取出来

在挑选适用的Advisor时,BeanFactoryTransactionAttributeSourceAdvisor间接实现了PointcutAdvisor,因此会进入 canApply 函数中的第二个 if 判断。会将BeanFactoryTransactionAttributeSourceAdvisor中的getPointcut()方法的返回值作为参数继续调用 canApply 方法。

接着回去判断这个类或者类的方法上是否添加了@Transactional注解,如果有的话,继续调用parseTransactionAnnotation方法解析详细的属性

至此,就讲BeanFactoryTransactionAttributeSourceAdvisor注入了目标对象的拦截器链中,BeanFactoryTransactionAttributeSourceAdvisor的adviceBeanName属性中。创建代理的时候,会调用Advisor.getAdvice方法,此时就会得到的该增强方法TransactionInterceptor,因为TransactionInterceptor实现了Advice,将其作为 CGLIB 的 CallBack 信息,@Transactional修饰方法所在的类作为 CGLIB 的父类信息,CGLIB 根据这些信息生成对应的子类的 Class 对象作为代理类。所以,在调用事务增强器增强的代理类时,会首先执行TransactionInterceptor进行增强,同时,也就是在TransactionInterceptor类中的invoke方法中完成了整个事务的逻辑。

3.方法拦截

当程序执行事务方法的时候,就会先走增强器TransactionInterceptor.invoke方法:

public Object invoke(final MethodInvocation invocation) throws Throwable {
      
        Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null);

        return invokeWithinTransaction(invocation.getMethod(), targetClass, new InvocationCallback() {
            @Override
            public Object proceedWithInvocation() throws Throwable {
                return invocation.proceed();
            }
        });
    }

protected Object invokeWithinTransaction(Method method, Class<?> targetClass, final InvocationCallback invocation)
            throws Throwable {

        // 获取对应事务属性
        final TransactionAttribute txAttr = getTransactionAttributeSource().getTransactionAttribute(method, targetClass);
        // 获取 beanFactory 中的 transactionManager,需要配置
        final PlatformTransactionManager tm = determineTransactionManager(txAttr);
        // 获取方法唯一标识(类.方法 如 xxx.UserServiceImpl.save)
        final String joinpointIdentification = methodIdentification(method, targetClass, txAttr);

        // 声明式事务处理 @Transactional
        if (txAttr == null || !(tm instanceof CallbackPreferringPlatformTransactionManager)) {
            // 创建 TransactionInfo,开启事务
            TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification);
            Object retVal = null;
            try {
                // 执行被增强方法
                retVal = invocation.proceedWithInvocation();
            }
            catch (Throwable ex) {
                // 异常回滚
                completeTransactionAfterThrowing(txInfo, ex);
                throw ex;
            }
            finally {
                cleanupTransactionInfo(txInfo);
            }
            // 提交事务
            commitTransactionAfterReturning(txInfo);
            return retVal;
        }
        //省略编程式事务和异常处理

可以看到在该拦截器的方法实现中,对目标方法进行了类似于环绕通知对事务处理,由此就为该方法添加了事务功能

参考文献:https://www.jianshu.com/p/9fa0b3c17810

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@transactional注解的实现原理涉及到Spring框架的事务管理模块。下面是大致的源码分析: 1. @transactional注解的处理过程是由Spring的事务拦截器(TransactionInterceptor)完成的。该拦截器是一个AOP切面,会在被注解的方法调用前后进行拦截。 2. 当一个方法被@transactional注解修饰时,Spring会根据注解的属性进行解析,例如事务的隔离级别、传播行为等。 3. 在方法调用前,事务拦截器会创建一个事务上下文(TransactionContext),该上下文包含了事务的相关信息,如连接、事务隔离级别等。 4. 事务拦截器会根据事务上下文中的信息创建一个事务对象(TransactionObject),该对象包含了事务的状态和操作方法,如开启事务、提交事务、回滚事务等。 5. 在方法调用前,事务拦截器会检查当前线程是否存在有效的事务上下文。如果存在,则使用已有事务上下文;如果不存在,则创建一个新的事务上下文。 6. 在方法调用后,事务拦截器会根据方法的执行结果,决定是提交事务还是回滚事务。如果方法执行成功,则提交事务;如果方法执行失败,则回滚事务。 7. 当方法调用结束后,事务拦截器会清理事务上下文,释放相关资源。 以上是大致的@transactional注解实现源码分析,具体的实现细节还涉及到Spring框架的事务管理器、事务切面等组件。如果你对具体的源码实现感兴趣,可以参考Spring框架的源码
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值