文档章节

MyBatis-3.4.2-源码分析16:XML解析之SqlSessionFactory|SqlSession

强子大叔的码田
 强子大叔的码田
发布于 2017/04/06 19:32
字数 608
阅读 33
收藏 2

首先sqlSessionFactory的类为:

main[1] print sqlSessionFactory
 sqlSessionFactory = "org.apache.ibatis.session.defaults.DefaultSqlSessionFactory@3e2e18f2"
main[1] 

得知,常规情况下,生成了DefaultSqlSessionFactory类实例。

接下来,打开session

sqlSession = sqlSessionFactory.openSession();

我们来看看,发生了什么

@Override
	public SqlSession openSession() {
		//开始
		return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);
	}

---

configuration.getDefaultExecutorType()默认返回simple类型,

Step completed: "thread=main", org.apache.ibatis.session.Configuration.getDefaultExecutorType(), line=408 bci=0
408    		return defaultExecutorType;

main[1] print defaultExecutorType
 defaultExecutorType = "SIMPLE"

---继续执行openSessionFromDataSource函数

private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level,
			boolean autoCommit) {
		// 看到这里了
		// 初始化为nul
		Transaction tx = null;
		try {
			// 开始执行
			// 这里返回一个封装对象,里面有事务工厂句柄 & 连接池句柄
			final Environment environment = configuration.getEnvironment();
			// 取出事务工厂句柄
			// org.apache.ibatis.transaction.jdbc.JdbcTransactionFactory
			final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
			// 继续执行
			tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
			final Executor executor = configuration.newExecutor(tx, execType);
			return new DefaultSqlSession(configuration, executor, autoCommit);
		} catch (Exception e) {
			closeTransaction(tx); // may have fetched a connection so lets call
									// close()
			throw ExceptionFactory.wrapException("Error opening session.  Cause: " + e, e);
		} finally {
			ErrorContext.instance().reset();
		}
	}

解读:上面这段代码,是找到environment对象,取出事务工厂实例,

然后执行了一行很重要的代码:

tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);

下面,我们开始来仔细研究这行代码。

--------------------------------------------------------------------------------分割线

这里的dataSource我已经设置成了阿里的druid类型,打印如下:

main[1] print dataSource.getClass()
 dataSource.getClass() = "class com.alibaba.druid.pool.DruidDataSource"
main[1] 

继续执行newTransaction函数,代码如下,很简单

  @Override
  public Transaction newTransaction(DataSource ds, TransactionIsolationLevel level, boolean autoCommit) {
    return new JdbcTransaction(ds, level, autoCommit);
  }

先记住1个断点

stop in org.apache.ibatis.transaction.jdbc.JdbcTransactionFactory.newTransaction

返回的事务对象为

org.apache.ibatis.transaction.jdbc.JdbcTransaction

接下来,是执行

final Executor executor = configuration.newExecutor(tx, execType);

---继续处理

public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
		// 从这里开始看
		executorType = executorType == null ? defaultExecutorType : executorType;
		executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
		// 在这里,还是simple模式
		Executor executor;
		if (ExecutorType.BATCH == executorType) {
			executor = new BatchExecutor(this, transaction);
		} else if (ExecutorType.REUSE == executorType) {
			executor = new ReuseExecutor(this, transaction);
		} else {
			// 执行到这里
			executor = new SimpleExecutor(this, transaction);
		}
		// 生成了SimpleExecutor变量
		//我觉得这里不应该用cacheEnabled,所以我在settings里把它屏蔽掉了
		//接下来就不执行
		if (cacheEnabled) {
			executor = new CachingExecutor(executor);
		}
		executor = (Executor) interceptorChain.pluginAll(executor);
		return executor;
	}

然后,就是执行

executor = (Executor) interceptorChain.pluginAll(executor);

看字面,是interceptorChain添加这个plugin.我们看具体实现吧

public Object pluginAll(Object target) {
	  //从这里开始执行
	  //这里是插件机制了
    for (Interceptor interceptor : interceptors) {
      target = interceptor.plugin(target);
    }
    return target;
    //返回自己
  }

如果后面有需求,就应该在这定制插件plugin.

---------------------------------------------------继续执行

return new DefaultSqlSession(configuration, executor, autoCommit);

这个没啥好说的,总结一下:

生成了一个DefaultSqlSession对象,这个对象包含executor,

executor包含transaction对象,transaction对象包含dataSource对象。

 

© 著作权归作者所有

强子大叔的码田

强子大叔的码田

粉丝 910
博文 1439
码字总数 1221048
作品 9
南京
架构师
私信 提问
Mybatis源码概览(一)

一般拿到源码会无从下手,我的基本思路一般就是根据一个基本的helloWorld Debug下去,把主线先大概理一遍,然后再具体分析细节,没有必要一个类一个类细看,看了也会忘掉。自己理源码的时候看...

robin-yao
2016/03/22
872
0
【深入浅出MyBatis系列一】MyBatis入门

深入浅出MyBatis系列 【深入浅出MyBatis系列一】MyBatis入门 【深入浅出MyBatis系列二】配置简介(MyBatis源码篇) 【深入浅出MyBatis系列三】Mapper映射文件配置 【深入浅出MyBatis系列四】...

陶邦仁
2015/12/22
6K
0
Mybatis执行流程(一)

搭建一个简单的Mybatis+Maven项目 Maven依赖 代码 配置文件 通过SqlSessionFactory获取SqlSession 执行流程图 源码走读 获取reader 获取SqlSessionFactory build()方法 解析配置返回configu...

wjk_snail
2016/06/22
116
0
Mybatis 源码(一)总揽

整体架构 这只是MySql的一个逻辑划分架构。 接口层:通类提供对数据库访问能力,隐藏了后续复杂的处理逻辑。 核心处理层:主要负责执行SQL,并返回结果。 基础支撑层:对一些基础功能进行封装...

xiaolyuh
前天
44
0
Mybatis源码概览(二) ---Plugin扩展与Spring结合原理

本文主要介绍Mybatis通过动态代理避免对sqlSession直接调用,而是通过MapperProxy代理技术生成了具体dao接口的Mapper实例,里面封装了对sqlSession的调用;Mybatis预留了Interceptor接口,用...

robin-yao
2016/03/23
696
0

没有更多内容

加载失败,请刷新页面

加载更多

PostgreSQL 11.3 locking

rudi
今天
5
0
Mybatis Plus sql注入器

一、继承AbstractMethod /** * @author beth * @data 2019-10-23 20:39 */public class DeleteAllMethod extends AbstractMethod { @Override public MappedStatement injectMap......

一个yuanbeth
今天
10
1
一次写shell脚本的经历记录——特殊字符惹的祸

本文首发于微信公众号“我的小碗汤”,扫码文末二维码即可关注,欢迎一起交流! redis在容器化的过程中,涉及到纵向扩pod实例cpu、内存以及redis实例的maxmemory值,statefulset管理的pod需要...

码农实战
今天
4
0
为什么阿里巴巴Java开发手册中不建议在循环体中使用+进行字符串拼接?

之前在阅读《阿里巴巴Java开发手册》时,发现有一条是关于循环体中字符串拼接的建议,具体内容如下: 那么我们首先来用例子来看看在循环体中用 + 或者用 StringBuilder 进行字符串拼接的效率...

武培轩
今天
8
0
队列-链式(c/c++实现)

队列是在线性表功能稍作修改形成的,在生活中排队是不能插队的吧,先排队先得到对待,慢来得排在最后面,这样来就形成了”先进先出“的队列。作用就是通过伟大的程序员来实现算法解决现实生活...

白客C
今天
81
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部