事务环境搭建工作链接
注解 @EnableTransactionManagement:在配置类中添加注解@EnableTransactionManagement,便开启了事务功能。此注解也是了解Spring事务源码的入口。

@EnableTransactionManagement
public class TxConfig {

【1】@EnableTransactionManagement 利用 TransactionManagementConfigurationSelector 给容器中导入组件:如下代码所示,SelectorImposrts共导入了两个组件:①、AutoProxyRegistrar、②、ProxyTransactionManagementConfiguration

// xxxSelector 继承自 ImportSelector 表示给容器中导入组件
@Import({TransactionManagementConfigurationSelector.class})
public @interface EnableTransactionManagement {......} //进入TransactionManagementConfigurationSelector的selectImports方法,发现导入两个组件
public class TransactionManagementConfigurationSelector extends AdviceModeImportSelector<EnableTransactionManagement> {
protected String[] selectImports(AdviceMode adviceMode) {
switch(null.$SwitchMap$org$springframework$context$annotation$AdviceMode[adviceMode.ordinal()]) {
case 1:
//导入的两个组件:AutoProxyRegistrar、ProxyTransactionManagementConfiguration
return new String[]{AutoProxyRegistrar.class.getName(), ProxyTransactionManagementConfiguration.class.getName()};
}
}

【2】组件一:AutoProxyRegistrar 给容器中注册了一个 InfrastructureAdvisorAutoProxyCreator 组件。该组件继承自xxxAwareBeanPostProcessor后置处理器,利用后置处理器机制在对象创建以后进行包装,返回一个代理对象(有增强器),代理对象通过拦截器链进行调用。

//通过继承类《BeanDefinitionRegister》可以猜到该类的功能,主要是给容器中注册Bean的属性
public class AutoProxyRegistrar implements ImportBeanDefinitionRegistrar {
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
//......
AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);
}
} //进入上面的registerxxx方法:创建一个自动代理创建器
public static BeanDefinition registerAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry) {
return registerAutoProxyCreatorIfNecessary(registry, (Object)null);
} //进入上面registerxxx 发现它给容器中注册了一个 InfrastructureAdvisorAutoProxyCreator 组件
public static BeanDefinition registerAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, @Nullable Object source) {
return registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
}

【3】ProxyTransactionManagementConfiguration 是一个配置类,如下:
  第一步:给容器中注册事务增强器:1)、主要用于对事务注解 @Transactional 中的属性进行解析。给容器中注入:AnnotationTransactionAttributeSource 实例。

@Configuration(
proxyBeanMethods = false
)
public class ProxyTransactionManagementConfiguration extends AbstractTransactionManagementConfiguration {
//注册事务增强器
@Bean(
name = {"org.springframework.transaction.config.internalTransactionAdvisor"}
)
@Role(2)
public BeanFactoryTransactionAttributeSourceAdvisor transactionAdvisor(TransactionAttributeSource transactionAttributeSource, TransactionInterceptor transactionInterceptor) {
BeanFactoryTransactionAttributeSourceAdvisor advisor = new BeanFactoryTransactionAttributeSourceAdvisor();
//设置一个事务属性
advisor.setTransactionAttributeSource(transactionAttributeSource);
//设置一个事务拦截器
advisor.setAdvice(transactionInterceptor);
if(this.enableTx != null) {
advisor.setOrder(((Integer)this.enableTx.getNumber("order")).intValue());
} return advisor;
} //事务属性源码展示:也是一个bean,注入到容器中
@Bean
@Role(2)
public TransactionAttributeSource transactionAttributeSource() {
return new AnnotationTransactionAttributeSource();
} //注入的bean对象源码展示:主要功能使用注解解析器对事务注解进行解析 publicMethodsOnly=true
public AnnotationTransactionAttributeSource(boolean publicMethodsOnly) {
this.publicMethodsOnly = publicMethodsOnly;
this.annotationParsers = new LinkedHashSet(4);
//事务注解解析器
this.annotationParsers.add(new SpringTransactionAnnotationParser());
if(jta12Present) {
this.annotationParsers.add(new JtaTransactionAnnotationParser());
} if(ejb3Present) {
this.annotationParsers.add(new Ejb3TransactionAnnotationParser());
}
}
} //进入事务注解解析器 SpringTransactionAnnotationParser 中的parseTransactionAnnotation方法
//里面的都是都是可以在@Transactional中配置的属性
protected TransactionAttribute parseTransactionAnnotation(AnnotationAttributes attributes) {
RuleBasedTransactionAttribute rbta = new RuleBasedTransactionAttribute();
Propagation propagation = (Propagation)attributes.getEnum("propagation");
rbta.setPropagationBehavior(propagation.value());
Isolation isolation = (Isolation)attributes.getEnum("isolation");
rbta.setIsolationLevel(isolation.value());
rbta.setTimeout(attributes.getNumber("timeout").intValue());
rbta.setReadOnly(attributes.getBoolean("readOnly"));
rbta.setQualifier(attributes.getString("value"));
List<RollbackRuleAttribute> rollbackRules = new ArrayList();
Class[] var6 = attributes.getClassArray("rollbackFor");
int var7 = var6.length;
//......
}

2)、事务拦截器 :TransactionInterceptor 继承自 MethodInterceptor【方法拦截器:目前给容器中放置了一个代理对象,代理对象要执行目标方法时,方法拦截器就进行工作】 ,保存了事务属性信息和事务管理器;在目标方法执行的时候,代理对象会执行拦截器链【TransactionInterceptor 】,TransactionInterceptor(事务拦截器) 的作用:①、获取事务相关属性;②、获取平台事务管理器【platformTransactionManager 】获取方式:首先会根据 Transaction 中定义的 Qualifier 属性获取事务管理器,如果未定义则根据 TransactionManagerBeanName 获取,如果事先没有指定,则根据自动装配的值获取,如果没有指定,最终会从容器中按照类型获取一个 PlatformTransactionManager;③、开启事务、执行目标方法、如果发生异常,利用事务管理器回滚操作。如果正常,则利用事务管理器提交事务。

//继承了方法拦截器
public class TransactionInterceptor extends TransactionAspectSupport
implements MethodInterceptor, Serializable { //拦截器工作通过调用invoke进行触发。
@Nullable
public Object invoke(MethodInvocation invocation) throws Throwable {
Class<?> targetClass = invocation.getThis() != null?AopUtils.getTargetClass(invocation.getThis()):null;
Method var10001 = invocation.getMethod();
invocation.getClass();
//******重点,通过事务进行工作**** 进入该方法
return this.invokeWithinTransaction(var10001, targetClass, invocation::proceed);
} /**
* 事务拦截器的作用
*/
@Nullable
protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass, TransactionAspectSupport.InvocationCallback invocation) throws Throwable {
//省略部分非重要代码
//获取事务的属性
TransactionAttribute txAttr = tas != null?tas.getTransactionAttribute(method, targetClass):null;
//获取事务的管理器 ,附源码
PlatformTransactionManager tm = this.determineTransactionManager(txAttr);
//得到要执行的事务方法
String joinpointIdentification = this.methodIdentification(method, targetClass, txAttr);
//开启事务
TransactionAspectSupport.TransactionInfo txInfo = this.createTransactionIfNecessary(tm, txAttr, joinpointIdentification); try { //invocation 事务方法 下面是执行事务方法
retVal = invocation.proceedWithInvocation();
} catch (Throwable var17) {
//进入源码,发现是通过事务管理器进行回滚
this.completeTransactionAfterThrowing(txInfo, var17);
throw var17;
} finally {
this.cleanupTransactionInfo(txInfo);
} //......
//如果正常执行,则提交事务
this.commitTransactionAfterReturning(txInfo);
return retVal;
} //获取事务管理器的源码,首先会根据Transaction中定义的qualifier值获取管理器
@Nullable
protected PlatformTransactionManager determineTransactionManager(@Nullable TransactionAttribute txAttr) {
if(txAttr != null && this.beanFactory != null) {
String qualifier = txAttr.getQualifier();
if(StringUtils.hasText(qualifier)) {
return this.determineQualifiedTransactionManager(this.beanFactory, qualifier); //如果qualifier 是空的则根据 transactionManagerBeanName
} else if(StringUtils.hasText(this.transactionManagerBeanName)) {
return this.determineQualifiedTransactionManager(this.beanFactory, this.transactionManagerBeanName);
} else { //如果都没指定,则根据自动装配的值获取
PlatformTransactionManager defaultTransactionManager = this.asPlatformTransactionManager(this.getTransactionManager());
if(defaultTransactionManager == null) {
defaultTransactionManager = this.asPlatformTransactionManager(this.transactionManagerCache.get(DEFAULT_TRANSACTION_MANAGER_KEY));
//这是我们一般会最终调用的地方,会按照类型获取事务管理器 PlatformTransactionManager
if(defaultTransactionManager == null) {
defaultTransactionManager = (PlatformTransactionManager)this.beanFactory.getBean(PlatformTransactionManager.class);
this.transactionManagerCache.putIfAbsent(DEFAULT_TRANSACTION_MANAGER_KEY, defaultTransactionManager);
}
}
return defaultTransactionManager;
}
}
}
}

 ----关注公众号,获取更多内容----

Spring 事务——源码分析的更多相关文章

  1. [心得体会]spring事务源码分析

    spring事务源码分析 1. 事务的初始化注册(从 @EnableTransactionManagement 开始) @Import(TransactionManagementConfigurati ...

  2. spring事务源码分析结合mybatis源码(一)

    最近想提升,苦逼程序猿,想了想还是拿最熟悉,之前也一直想看但没看的spring源码来看吧,正好最近在弄事务这部分的东西,就看了下,同时写下随笔记录下,以备后查. spring tx源码分析 这里只分析 ...

  3. Spring事务源码分析专题(一)JdbcTemplate使用及源码分析

    Spring中的数据访问,JdbcTemplate使用及源码分析 前言 本系列文章为事务专栏分析文章,整个事务分析专题将按下面这张图完成 对源码分析前,我希望先介绍一下Spring中数据访问的相关内容 ...

  4. spring事务源码分析结合mybatis源码(三)

    下面将结合mybatis源码来分析下,这种持久化框架是如何对connection使用,来达到spring事务的控制. 想要在把mybatis跟spring整合都需要这样一个jar包:mybatis-s ...

  5. Spring事务源码分析总结

    Spring事务是我们日常工作中经常使用的一项技术,Spring提供了编程.注解.aop切面三种方式供我们使用Spring事务,其中编程式事务因为对代码入侵较大所以不被推荐使用,注解和aop切面的方式 ...

  6. Spring事务源码分析

    首先看例子,这例子摘抄自开涛的跟我学spring3. @Test public void testPlatformTransactionManager() { DefaultTransactionDe ...

  7. spring事务源码分析结合mybatis源码(二)

    让我们继续上篇,分析下如果有第二个调用进入的过程. 代码部分主要是下面这个: if (isExistingTransaction(transaction)) { return handleExisti ...

  8. spring事务源码研读1

    转载摘录自:Spring事务源码分析(一)Spring事务入门 有时为了保证一些操作要么都成功,要么都失败,这就需要事务来保证. 传统的jdbc事务如下: @Test public void test ...

  9. 框架源码系列十一:事务管理(Spring事务管理的特点、事务概念学习、Spring事务使用学习、Spring事务管理API学习、Spring事务源码学习)

    一.Spring事务管理的特点 Spring框架为事务管理提供一套统一的抽象,带来的好处有:1. 跨不同事务API的统一的编程模型,无论你使用的是jdbc.jta.jpa.hibernate.2. 支 ...

  10. Spring AMQP 源码分析 02 - CachingConnectionFactory

    ### 准备 ## 目标 了解 CachingConnectionFactory 在默认缓存模式下的工作原理   ## 前置知识   <Spring AMQP 源码分析 01 - Impatie ...

随机推荐

  1. 微信小程序云函数中管理短信验证码的完整SDK工具

    微信小程序云开发管理短信验证码并不是一件容易的事情,它不像其他web服务器开发中可以将验证码存放到session或者cookie中,你只能将其存到云开发的数据库中,实现起来非常困难.一个简单的需要短信 ...

  2. javascript向tabale中动态添加数据

    <table width="600" border="1" cellspacing="0"> <thead> < ...

  3. UnsupportedOperationException异常

    看看下面的例子,这样输出什么呢? public class test { public static void main(String[] args) { String arr = "ab, ...

  4. 方法(Java)

    什么是方法? 基本介绍 在其他语言中也叫函数 System.out.println();类名.对象.方法: Java方法是语句的集合,它们在一起执行一个功能 方法是解决一类问题的步骤的有序集合 方法包 ...

  5. AntD为Form的List设置默认值 (antd form.list 设置默认值 )

    import React from "react"; function demo() { const FormConfig = { labelCol: { span: 8 }, w ...

  6. PR2022(Premiere Pro 2022)Mac/win最新中文版

    Adobe Premiere Pro 2022 Mac/win是用于专业级别的视频编辑软件,一个基于时间轴的视频处理工具,具有许多用于生成高端视频的不同功能.Adobe Premiere Pro最重要 ...

  7. Python抓取数据具体流程

    之前看了一段有关爬虫的网课深有启发,于是自己也尝试着如如何过去爬虫百科"python"词条等相关页面的整个过程记录下来,方便后期其他人一起来学习. 抓取策略 确定目标:重要的是先确 ...

  8. python更新pip报错pip._vendor.urllib3.exceptions.ProxySchemeUnknown: Not supported proxy scheme None

    更新pip报错: 看到最后一行很明显是proxy的问题,查看cmd下的代理 将代理删掉重启cmd执行命令就不会报错了

  9. 【转载】Python 在已存在的excel表格中写入数据

    由于每天都要汇总日报数据,其实就是个复制粘贴的工作,将多个表的数据汇总成一个表格,工作较枯燥,于是想到用Python帮我完成这个简单又粗糙的工作.写一下我学到的几个小技巧,下次忘记怎么使用的时候,还能 ...

  10. 07.异常、多线程、Lambda 表达式

    一.异常 指的是程序在执行过程中,出现的非正常的情况,最终会导致JVM的非正常停止. 异常体系 根类 java.lang.Throwable 两个直接子类 java.lang.Error 严重错误Er ...