spring事务管理源码解析--加了@Transactional注解后Spring究竟为我们做了哪些事情?
大家都知道事务管理是基于AOP的,对AOP还不了解的请自行百度。
实现一个事务需要以下几步:1.获取数据库连接 2.执行数据库操作 3.如果2步骤发生异常就回滚,否则就提交 4.释放资源。
然后1、3、4步骤是所有事务所共有的逻辑,程序员真正需要关心的只有第2步,spring的事务管理也正是帮我们做了1、3、4的工作。
那么,我们想在项目中使用spring事务需要做哪些准备呢?
第一步,要配置dataSource、transactionManager
<!--启动spring注解功能 -->
<tx:annotation-driven transactionmanager="transactionManager" /> <!-- 设定transactionManager -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean> <!-- 配置数据源,这里以C3P0为例 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
destroy-method="close">
<property name="driverClass" value="..."></property>
...
<property name="checkoutTimeout" value="..."/>
</bean>
第二步,在需要事务的方法上加上注解 @Transactional 即可。
是的,就是这么简单。下面通过分析源码来说明整个流程,我们先来看配置这个注解时可以设置哪些参数。

这些配置一般我们只关注传播属性、隔离级别和超时时间就行,具体不懂得自行百度。
我们再回过头来看前面配置的transactionManager(即DataSourceTransactionManager)继承关系图
我们只看PlatformTransactionManager-->AbstractPlatformTransactionManager-->DataSourceTransactionManager这条线,先来看PlatformTransactionManager接口:

它是spring事务管理器的顶层接口,这三个方法是对应前边所说的1、3两个步骤,分别为开启事务、提交、回滚。再来看它的实现类AbstractPlatformTransactionManager,主要看上边三个方法对应的子类实现,可以发现它们都是一样的套路:先做一些边界条件判断,然后调用自己内部的对应do...方法,而且每个对应的do...方法都是protected abstract 修饰的,实际上继承于AbstractPlatformTransactionManager 的类有很多,如DataSourceTransactionManager , JtaTransactionManager, HibernateTransactionManager 等,不管是哪一个种,总要有 getTransaction,commit,rollback的方法,所以这个三个方法在接口中,不过具体实现方式有所区别,所以具体的实现需要而且是必须在子类中完成。AbstractPlatformTransactionManager 规定了这三个方法的调用顺序,真正的细节在子类中以”do*”开头的方法中实现,这也正是大多数抽象类的作用所在。
我们先来看此抽象类的getTransaction方法:

重点关注我标注出来的两步,先调用doGetTransaction方法获取一个事务(新开启的或者已经存在的),然后调用doBegin方法开始执行事务。
我们直接去看具体实现类的这两个方法,这里以DataSourceTransactionManager为例。doGetTransaction方法其实就是返回我们配置的全局数据源。
再来看doBegin方法,先获取一个connetion,设置autoCommit为false,设置隔离级别...

然后调用 service 中的代码,如果没有抛出异常Spring框架将继续调用AbstractPlatformTransactionManager中的commit方法,继续看源码:


还是要看子类doCommit方法,先拿到连接(其实就是之前获取到的那个),然后commit,有异常就抛给上层。
这里先抛出两个问题:spring事务管理如何保证同一个线程总是能获取到已开启的事务?也就是说默认传播属性下同一个线程怎么保证最多只会有一个事务?

现在事务已经提交了,如果发生异常的话,父类中调用方法 rollback方法:

继续看子类doRollback方法:

现在回想我刚才抛出的两个问题,不知道你有没有发现,在提交和回滚的是获取当前连接都是这样调用:
Connection con = txObject.getConnectionHolder().getConnection();
那怎么保证每次get的是同一个连接?
来看的dobegin方法,重点关注bind the session holder这个方法

附上resource的定义,可以发现它是一个ThreadLocal。
private static final ThreadLocal<Map<Object, Object>> resources = new NamedThreadLocal<Map<Object, Object>>("Transactional resources");
现在明白了吧,spring在开启事务的时候会把获取到的连接绑定在事务的上下文环境中,然后把这个上下文环境又绑定在当前线程的ThreadLocal中,这样每个线程的事务就是独立的了,那同一个线程每次获取连接获取到的也就是同一个了。
最后看一下对资源释放的代码,父类中cleanupAfterCompletion方法调用了doCleanupAfterCompletion,我们直接看子类实现:

releaseConnection方法就是进行资源释放的工作,逻辑比较简单就不分析了。
至此 Spring 帮我们管理的事务,其主要流程和方法已经介绍完了,当然其中还有很多 private 方法和 protected 方法没有介绍。我想只要把主干捋清楚之后,其他一些方法也很好理解了。
绕了一大圈,实际上还是绕不过开启事务,提交事务,回滚事务三件事,只是这三件事情现在由Spring帮助我们在背后默默做好了。
spring事务管理源码解析--加了@Transactional注解后Spring究竟为我们做了哪些事情?的更多相关文章
- Spring 与 MyBatis 事务管理源码解析
		
用到mybatis便由spring和myabtis集成,SqlSessionFactoryBean(直接负责对mybatis所需环境的创建) ,配置相应的datasource到springConfig ...
 - Spring事务管理源码分析
		
Spring事务管理方式 依据Spring.xsd文件可以发现,Spring提供了advice,annotation-driven,jta-transaction-manager3种事务管理方式.详情 ...
 - Spring Security 解析(七) —— Spring Security Oauth2 源码解析
		
Spring Security 解析(七) -- Spring Security Oauth2 源码解析 在学习Spring Cloud 时,遇到了授权服务oauth 相关内容时,总是一知半解,因 ...
 - spring boot @Value源码解析
		
Spring boot 的@Value只能用于bean中,在bean的实例化时,会给@Value的属性赋值:如下面的例子: @SpringBootApplication @Slf4j public c ...
 - Feign 系列(05)Spring Cloud OpenFeign 源码解析
		
Feign 系列(05)Spring Cloud OpenFeign 源码解析 [TOC] Spring Cloud 系列目录(https://www.cnblogs.com/binarylei/p/ ...
 - [源码解析] 深度学习分布式训练框架 horovod (3) --- Horovodrun背后做了什么
		
[源码解析] 深度学习分布式训练框架 horovod (3) --- Horovodrun背后做了什么 目录 [源码解析] 深度学习分布式训练框架 horovod (3) --- Horovodrun ...
 - iOS开发SDWebImage源码解析之SDWebImageManager的注解
		
最近看了两篇博客,写得很不错,关于SDWebImage源码解析之SDWebImageManager的注解: 1.http://www.jianshu.com/p/6ae6f99b6c4c 2.http ...
 - Spring注解Component原理源码解析
		
在实际开发中,我们经常使用Spring的@Component.@Service.@Repository以及 @Controller等注解来实现bean托管给Spring容器管理.Spring是怎么样实 ...
 - 【Spring实战】Spring注解配置工作原理源码解析
		
一.背景知识 在[Spring实战]Spring容器初始化完成后执行初始化数据方法一文中说要分析其实现原理,于是就从源码中寻找答案,看源码容易跑偏,因此应当有个主线,或者带着问题.目标去看,这样才能最 ...
 
随机推荐
- C# 连接SQL数据库以及操作数据库
			
1.概述 ado.net提供了丰富的数据库操作,这些操作可以分为三个步骤: 第一,使用SqlConnection对象连接数据库: 第二,建立SqlCommand对象,负责SQL语句的执行和存储过程的调 ...
 - ubuntu14.04 安装 pyv8
			
1. $sudo pip install -v pyv8 Error: pip unicodedecodeerror ‘ascii’ codec can’t decode byte 0xe2 in p ...
 - HDU 1181.变形课-并查集
			
变形课 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others)Total Submis ...
 - 洛谷P1095 绝地武士的逃离
			
好吧原题是守望者的逃离,我强行改了一波题面,因为信仰=-=(? May the force be with us. 绝地跑步速度为17m/s,但无法逃离荒岛.绝地的原力恢复速度为4点/s,只有处在原地 ...
 - [BZOJ 2342] 双倍回文
			
Link:https://www.lydsy.com/JudgeOnline/problem.php?id=2342 Algorithm: 解决回文串问题,一般从对称轴下手. 肯定先跑一边Manach ...
 - [CF494D]Birthday
			
题意:给一棵带边权的树,定义如下的一些东西 $S(x)$表示以$x$为根的子树中的节点组成的集合 $d(u,v)$表示$u$和$v$之间的距离 $f(u,v)\sum\limits_{x\in S(v ...
 - 【并查集+离散化】BZOJ4195- [Noi2015]程序自动分析
			
[题目大意] 在实现程序自动分析的过程中,常常需要判定一些约束条件是否能被同时满足. 考虑一个约束满足问题的简化版本:假设x1,x2,x3,…代表程序中出现的变量,给定n个形如xi=xj或xi≠xj的 ...
 - 【欧拉函数】BZOJ2190-[SDOI2012]longge的数学问题
			
[题目大意] 求出∑gcd(i, N)(1<=i <=N). [思路] 对于x=ak,y=bk,若gcd(a,b)=1则必有gcd(x,y)=1.枚举N的所有因数,∑gcd(i, N)=∑ ...
 - 动态NAT地址转换
			
1.配置路由器的端口ip地址(注意外网和内网ip地址的设置) Router(config)#inter f0/0 Router(config-if)#ip add 192.168.1.1 255.25 ...
 - 冒泡排序--注意flag变量的设置
			
代码: #include<stdio.h> void BubbleSort(int a[],int n){ int i,j; int temp; ; // 此处flag变量的设置可以提高算 ...