SpringBoot事务原理解析

Spring中使用事务需要添加注解,表示开启事务

@EnableTransactionManagement

这个注解有个默认属性

AdviceMode mode() default AdviceMode.PROXY;

说明默认是使用JDK代理的通知模式。

然后注解上面使用了@Import注解

@Import(TransactionManagementConfigurationSelector.class)

我们再来看TransactionManagementConfigurationSelector这个类。

    @Override
	protected String[] selectImports(AdviceMode adviceMode) {
		switch (adviceMode) {
			case PROXY:
				return new String[] {AutoProxyRegistrar.class.getName(),
						ProxyTransactionManagementConfiguration.class.getName()};
			case ASPECTJ:
				return new String[] {determineTransactionAspectClass()};
			default:
				return null;
		}
	}

这个selectImports()方法返回了ProxyTransactionManagementConfiguration和AutoProxyRegistrar的全路径名,其中ProxyTransactionManagementConfiguration设置了

BeanFactoryTransactionAttributeSourceAdvisor2个属性TransactionInterceptor和TransactionAttributeSource,其中TransactionAttributeSource的实现类是AnnotationTransactionAttributeSource()。实例化的时候还指定了解析类是SpringTransactionAnnotationParser。

核心原理就是:SpringTransactionAnnotationParser解析@Transactional注解的类,将属性设置到TransactionAttributeSource中,然后调用方法时由拦截器TransactionInterceptor拦截实现事务的代理,使用PlatformTransactionManager来管理事务

因为我标注在@Controller上了,调用顺序如下

--》AbstractAutowireCapableBeanFactory#createBean(java.lang.String, org.springframework.beans.factory.support.RootBeanDefinition, java.lang.Object[])

--》initializeBean(beanName, exposedObject, mbd)

---》applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName)

--》Object current = processor.postProcessAfterInitialization(result, beanName);

--》wrapIfNecessary(bean, beanName, cacheKey)

--》Object proxy = createProxy( bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));//这里的specificInterceptors就是BeanFactoryTransactionAttributeSourceAdvisor对象。

--》return proxyFactory.getProxy(getProxyClassLoader());

 这个就是返回的代理类了,调用这个方法的时候直接就是使用了代理类的方法。

当我调用@Transactional标注的接口时,会调用org.springframework.aop.framework.CglibAopProxy.DynamicAdvisedInterceptor#intercept方法。

--》org.springframework.aop.framework.ReflectiveMethodInvocation#proceed

--》return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this)

--》org.springframework.transaction.interceptor.TransactionInterceptor#invoke

因为我用了MySQL数据库,连接池是Hikari连接池,这里的事务管理器是DataSourceTransactionManager,该管理器实现了PlatformTransactionManager接口。

接下来是真正的事务管理了

// This is an around advice: Invoke the next interceptor in the chain.
// This will normally result in a target object being invoked.
retVal = invocation.proceedWithInvocation();

如果抛异常

completeTransactionAfterThrowing(txInfo, ex);

然后finally

cleanupTransactionInfo(txInfo);

如果成功了

commitTransactionAfterReturning(txInfo);

底层是

txInfo.getTransactionManager().commit(txInfo.getTransactionStatus());

txInfo.getTransactionManager().rollback(txInfo.getTransactionStatus());

总结:

多看看TransactionAspectSupport#invokeWithinTransaction方法,这里是Spring的事务管理器的逻辑

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

飞翔的咩咩

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

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

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

打赏作者

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

抵扣说明:

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

余额充值