Mybatis整合Spring实现事务管理的源码分析
一:前言
没有完整看完,但是看到了一些关键的地方,这里做个记录,过程会有点乱,以后逐渐补充最终归档为完整流程;相信看过框架源码的都知道过程中无法完全确定是怎样的流程,毕竟不可能全部都去测试一遍
,但是看的过程中灵感的显现很重要(就是直觉知道接下来的步骤是什么应该是什么之类的,但是这个自觉是猜的而非蒙的,即过程里是有往会递推看到了一些关键点的而不是抛色子来确定是哪个子类)
,因此自己写的时候也无法将看的过程里产生的想法写得很细,过程也有点跳,如果大家有疑问最好自己去验证(方式就是搜索然后看哪里调用了这样一步步看下去就行,有些小技巧就是如果是public那么一般都是由外部类调用[入口],如果是protected却没实现
说明在之类里有实现,如果已经实现说明此方法在父类里会被调用,如果是private那么肯定在当前类会被调用);
二:重要类或方法
1)对于Mybatis
1.org.apache.ibatis.executor.Executor接口,有大概BatchExecutor/ReuseExecutor/SimpleExecutor三个子类有共同的父类BaseExecutor,它是DefaultSqlSession类的组件,sqlSession对象的如selectOne实际上是由executor执行的;
2.org.apache.ibatis.transaction.Transaction接口,它的实现类有多个,但是每个实现类里都有DataSource属性用于产生数据源和数据库服务交互;
3.SpringManagedTransaction是支持Spring管理Mybatis事务的关键;
4.DefaultSqlSessionFactory
5.DefaultSqlSession
它们之间的重要关系:sqlsession里有executor;executor里有transaction对象;transaction里有datasource;
流程:sqlsession执行代码委托为executor执行,executor里通过transaction获得connection;而transaction又通过判断threadlocalmap里是否有ConnectionHolder对象;如果没有则通过datasource获得一个,否则返回存在threadlocalmap里的;
三:执行流程
如:sqlSession.selectOne执行流程:
3.1selectOne最终转到selectList;
3.2调用sqlsession的属性executor.query方法执行;
3.3在BaseExecutor的query方法里执行这一句:list = this.queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);,因此这一句代码很关键;
3.4接着看里面有this.doQuery(ms, parameter, rowBounds, resultHandler, boundSql);方法;
3.5接着看SimpleExecutor的doQuery方法(看源码很多时候是靠即时的灵感,比较难完整描述看时的思路和过程);
补充开始:1.接着看stmt = this.prepareStatement(handler, ms.getStatementLog());,很重要,因为statement执行sql语句也是通过它内部的connection对象来执行的;
2.看Connection connection = this.getConnection(statementLog);,接着看里面有:Connection connection = this.transaction.getConnection(); 重要重要,transaction终于在执行sql的过程里出现了;
3.整合Spring的情况下通过本文的上下文知道此transaction是SpringManagedTransaction对象;
4.接着看SpringManagedTransaction的this.openConnection();有代码:
this.connection = DataSourceUtils.getConnection(this.dataSource);
this.autoCommit = this.connection.getAutoCommit();
this.isConnectionTransactional = DataSourceUtils.isConnectionTransactional(this.connection, this.dataSource);
5.我们看DataSourceUtils.getConnection(this.dataSource);,然后转到doGetConnection方法里有代码:
ConnectionHolder conHolder = (ConnectionHolder)TransactionSynchronizationManager.getResource(dataSource);
if(conHolder == null ...)
{通过dataSource获得一个新的connection}
else{
conHolder.getConnection(); //return一个在threadlocalmap里的connection对象;
}
注意,上上面的DataSourceUtils和这里当TransactionSynchronizationManager及ConnectionHolder是spring里的类不是mybatis的,里面的threadlocal对象都是静态的,因此mybatis可以获得spring中放在threadlocalmap的connectionholder对象;
而spring是否产生connectionholder以及是否放到threadlocalmap里是根据txAdvice策略来的;
补充结束;
3.6接着看handler.query(stmt, resultHandler);然后可以挑RoutingStatementHandler的这个方法看;
3.7会委托给其它handler处理,这里可暂定为PreparedStatementHandler类;
3.8看到ps.execute();,后面没什么好看了就是jdk中通过stmt执行sql的方式,这里关键的是stmt里的connection是什么;
四:和Spring整合过程中重要步骤
1.配置了org.mybatis.spring.SqlSessionFactoryBean,默认情况下它内部的transactionFactory属性是SpringManagedTransactionFactory对象(如果不想Spring来管理事务则可以主动指定sqlSessionFactoryBean的transactionFactory为别的类);
2.SqlSessionFactoryBean的sqlSessionFactory的configuration的environment里保存了transactionFactory对象为SpringManagedTransactionFactory对象,这一步很重要;
3.我们来看DefaultSqlSessionFactory的openSessionFromDataSource方法(真正获得sqlSession对象的地方),里面有代码
Environment environment = this.configuration.getEnvironment();
TransactionFactory transactionFactory = this.getTransactionFactoryFromEnvironment(environment);
tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
Executor executor = this.configuration.newExecutor(tx, execType);
var8 = new DefaultSqlSession(this.configuration, executor, autoCommit);
注意:execType默认对应的是SimpleExecutor,且这个executor里保存了SpringManagedTransactionFactory产生的SpringManagedTransaction对象,这里先明确一下;
4.上面已经明确了sqlSession执行语句是委托为exector来执行的;后面的步骤和三里的一致了就不赘述了;
五:Spring里事务管理的重要类和方法
1.org.springframework.transaction.interceptor.TransactionInterceptor,极其重要,这个是当执行pointcut方法时将会由这个事务拦截器根据<tx:attributes>配置的propagation做一些额外处理(是一个around),包括是否需要产生connection对象和将这个connection对象的autoCommit设置为什么值(SpringManagedTransaction的this.openConnection();会用到此值),比如REQUIRED就会先判断threadlocalmap里是否有connectionholder,有不做预处理没有则通过datasource获取connection并设置为autoCommit为false,然后通过
connectionholer包装此connection并set到threalocalmap里;
2.org.springframework.jdbc.datasource.DataSourceTransactionManager,极其重要,是spring用来根据propagation策略来执行不同的逻辑的重要组件(在TransactionInterceptor的invoke方法里执行);
3.TxNamespaceHandler用来处理名称空间为tx的元素;
4.TxAdviceBeanDefinitionParser用来解析<tx:advice元素,通过doParse处理;
5.NameMatchTransactionAttributeSource,极其重要,存储了<tx:attributes里配置的策略供TransactionInterceptor使用;
补充:如果要看懂spring的事务管理,可以从TransactionInterceptor的invoke方法实现开始看,这个是入口函数,而内部用到的ConnectionHolder则是和Mybatis有交互的类;
Mybatis整合Spring实现事务管理的源码分析的更多相关文章
- 【spring源码学习】spring的事务管理的源码解析
[一]spring事务管理(1)spring的事务管理,是基于aop动态代理实现的.对目标对象生成代理对象,加入事务管理的核心拦截器==>org.springframework.transact ...
- Spring Environment(二)源码分析
Spring Environment(二)源码分析 Spring 系列目录(https://www.cnblogs.com/binarylei/p/10198698.html) Spring Envi ...
- Spring Boot 揭秘与实战 源码分析 - 工作原理剖析
文章目录 1. EnableAutoConfiguration 帮助我们做了什么 2. 配置参数类 – FreeMarkerProperties 3. 自动配置类 – FreeMarkerAutoCo ...
- Spring Boot 揭秘与实战 源码分析 - 开箱即用,内藏玄机
文章目录 1. 开箱即用,内藏玄机 2. 总结 3. 源代码 Spring Boot提供了很多”开箱即用“的依赖模块,那么,Spring Boot 如何巧妙的做到开箱即用,自动配置的呢? 开箱即用,内 ...
- Spring笔记(5) - 声明式事务@EnableTransactionManagement注解源码分析
一.背景 前面详解了实现Spring事务的两种方式的不同实现:编程式事务和声明式事务,对于配置都使用到了xml配置,今天介绍Spring事务的注解开发,例如下面例子: 配置类:注册数据源.JDBC模板 ...
- spring事务概念与获取事务时事务传播行为源码分析
一.事务状态:org.springframework.transaction.TransactionStatus isNewTransaction 是否是新事务 hasSavepoint 是否有保存点 ...
- Spring Security(四) —— 核心过滤器源码分析
摘要: 原创出处 https://www.cnkirito.moe/spring-security-4/ 「老徐」欢迎转载,保留摘要,谢谢! 4 过滤器详解 前面的部分,我们关注了Spring Sec ...
- MyBatis(六):SqlSession执行源码分析
SqlSession执行源码分析 针对以下代码 public class MybatisUtils { private static SqlSessionFactory sqlSessionFacto ...
- Spring Cloud Eureka服务注册源码分析
Eureka是怎么work的 那eureka client如何将本地服务的注册信息发送到远端的注册服务器eureka server上.通过下面的源码分析,看出Eureka Client的定时任务调用E ...
随机推荐
- [知识梳理]课本1&2.1-2.5
面向对象的语言 出发点:更直接地描述客观世界中存在的事物(对象)以及它们之间的关系. 特点: 是高级语言. 将客观事物看作具有属性和行为的对象. 通过抽象找出同一类对象的共同属性和行为,形成类. 通过 ...
- C#之Socket通信
0.虽然之前在项目中也有用过Socket,但始终不是自己搭建的,所以对Server,Clinet端以及心跳,断线重连总没有很深入的理解,现在自己搭建了一遍加深一下理解. 服务端使用WPF界面,客户端使 ...
- PHP常见排序算法
$a = [1, 3, 5, 2, 4, 6, 12, 60, 45, 10, 32];$len = count($a);$num=0;/* * 冒泡排序 * 原理:不停的对相邻两个数进行比较,直到最 ...
- 新概念英语(1-63)Thank you, doctor.
新概念英语(1-63)Thank you, doctor. Who else is in bed today? why? A:How's Jimmy today? B:Better. Thank yo ...
- ssh整合之三hibernate和spring整合
1.拷贝我们的spring事务控制所需的jar包 2.在spring容器中配置我们的hibernateTemplate以及事务管理器 <?xml version="1.0" ...
- tomcat增加处理线程数量
修改server.xml <Executor name="tomcatThreadPool" namePrefix="catalina-exec-" ma ...
- Struts(十八):通过CURD来学习PrepareInterceptor拦截器
PrepareInterceptor拦截器的用法: 1.若Action实现了Preparable接口,则Action方法需实现prepare()方法: 2.PrepareInterceptor拦截器S ...
- html学习之多行文本
代码如下: <body> <form action="post"> <!--wrap设置多行文本是否自动换行--> <textarea n ...
- 1028阿里RDS如何恢复云数据库MySQL的备份文件到自建数据库
参照 https://help.aliyun.com/knowledge_detail/41817.html 恢复云数据库MySQL的备份文件到自建数据库 更新时间:2017-07-27 14:52: ...
- laypage分页控件使用方法
laypage是一款非常简单易用的分页控件,由于最近项目中使用到了,简单记录一下使用方法 1.引入laypage所需的js和css文件 <link href="js/laypage/1 ...