最近太忙了,一直没时间继续更新博客,今天忙里偷闲继续我的Mybatis学习之旅。在前九篇中,介绍了mybatis的配置以及使用, 那么本篇将走进mybatis的源码,分析mybatis 的执行流程, 好啦,鄙人不喜欢口水话,还是直接上干活吧:

1. SqlSessionFactory 与 SqlSession.

  通过前面的章节对于mybatis 的介绍及使用,大家都能体会到SqlSession的重要性了吧, 没错,从表面上来看,咱们都是通过SqlSession去执行sql语句(注意:是从表面看,实际的待会儿就会讲)。那么咱们就先看看是怎么获取SqlSession的吧:

(1)首先,SqlSessionFactoryBuilder去读取mybatis的配置文件,然后build一个DefaultSqlSessionFactory。源码如下:

  1. /**
  2. * 一系列的构造方法最终都会调用本方法(配置文件为Reader时会调用本方法,还有一个InputStream方法与此对应)
  3. * @param reader
  4. * @param environment
  5. * @param properties
  6. * @return
  7. */
  8. public SqlSessionFactory build(Reader reader, String environment, Properties properties) {
  9. try {
  10. //通过XMLConfigBuilder解析配置文件,解析的配置相关信息都会封装为一个Configuration对象
  11. XMLConfigBuilder parser = new XMLConfigBuilder(reader, environment, properties);
  12. //这儿创建DefaultSessionFactory对象
  13. return build(parser.parse());
  14. } catch (Exception e) {
  15. throw ExceptionFactory.wrapException("Error building SqlSession.", e);
  16. } finally {
  17. ErrorContext.instance().reset();
  18. try {
  19. reader.close();
  20. } catch (IOException e) {
  21. // Intentionally ignore. Prefer previous error.
  22. }
  23. }
  24. }
  25.  
  26. public SqlSessionFactory build(Configuration config) {
  27. return new DefaultSqlSessionFactory(config);
  28. }

(2)当我们获取到SqlSessionFactory之后,就可以通过SqlSessionFactory去获取SqlSession对象。源码如下:

  1. /**
  2. * 通常一系列openSession方法最终都会调用本方法
  3. * @param execType
  4. * @param level
  5. * @param autoCommit
  6. * @return
  7. */
  8. private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {
  9. Transaction tx = null;
  10. try {
  11. //通过Confuguration对象去获取Mybatis相关配置信息, Environment对象包含了数据源和事务的配置
  12. final Environment environment = configuration.getEnvironment();
  13. final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
  14. tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
  15. //之前说了,从表面上来看,咱们是用sqlSession在执行sql语句, 实际呢,其实是通过excutor执行, excutor是对于Statement的封装
  16. final Executor executor = configuration.newExecutor(tx, execType);
  17. //关键看这儿,创建了一个DefaultSqlSession对象
  18. return new DefaultSqlSession(configuration, executor, autoCommit);
  19. } catch (Exception e) {
  20. closeTransaction(tx); // may have fetched a connection so lets call close()
  21. throw ExceptionFactory.wrapException("Error opening session. Cause: " + e, e);
  22. } finally {
  23. ErrorContext.instance().reset();
  24. }
  25. }

通过以上步骤,咱们已经得到SqlSession对象了。接下来就是该干嘛干嘛去了(话说还能干嘛,当然是执行sql语句咯)。看了上面,咱们也回想一下之前写的Demo,

  1. SqlSessionFactory sessionFactory = null;
  2. String resource = "mybatis-conf.xml";
  3. try {
  4. //SqlSessionFactoryBuilder读取配置文件
  5. sessionFactory = new SqlSessionFactoryBuilder().build(Resources
  6. .getResourceAsReader(resource));
  7. } catch (IOException e) {
  8. e.printStackTrace();
  9. }
    //通过SqlSessionFactory获取SqlSession
    SqlSession sqlSession = sessionFactory.openSession();

还真这么一回事儿,对吧!

SqlSession咱们也拿到了,咱们可以调用SqlSession中一系列的select...,  insert..., update..., delete...方法轻松自如的进行CRUD操作了。 就这样? 那咱配置的映射文件去哪儿了?  别急, 咱们接着往下看:

2. 利器之MapperProxy:

在mybatis中,通过MapperProxy动态代理咱们的dao, 也就是说, 当咱们执行自己写的dao里面的方法的时候,其实是对应的mapperProxy在代理。那么,咱们就看看怎么获取MapperProxy对象吧:

(1)通过SqlSession从Configuration中获取。源码如下:

  1. /**
  2. * 什么都不做,直接去configuration中找, 哥就是这么任性
  3. */
  4. @Override
  5. public <T> T getMapper(Class<T> type) {
  6. return configuration.<T>getMapper(type, this);
  7. }

(2)SqlSession把包袱甩给了Configuration, 接下来就看看Configuration。源码如下:

  1. /**
  2. * 烫手的山芋,俺不要,你找mapperRegistry去要
  3. * @param type
  4. * @param sqlSession
  5. * @return
  6. */
  7. public <T> T getMapper(Class<T> type, SqlSession sqlSession) {
  8. return mapperRegistry.getMapper(type, sqlSession);
  9. }

(3)Configuration不要这烫手的山芋,接着甩给了MapperRegistry, 那咱看看MapperRegistry。 源码如下:

  1. /**
  2. * 烂活净让我来做了,没法了,下面没人了,我不做谁来做
  3. * @param type
  4. * @param sqlSession
  5. * @return
  6. */
  7. @SuppressWarnings("unchecked")
  8. public <T> T getMapper(Class<T> type, SqlSession sqlSession) {
  9. //能偷懒的就偷懒,俺把粗活交给MapperProxyFactory去做
  10. final MapperProxyFactory<T> mapperProxyFactory = (MapperProxyFactory<T>) knownMappers.get(type);
  11. if (mapperProxyFactory == null) {
  12. throw new BindingException("Type " + type + " is not known to the MapperRegistry.");
  13. }
  14. try {
  15. //关键在这儿
  16. return mapperProxyFactory.newInstance(sqlSession);
  17. } catch (Exception e) {
  18. throw new BindingException("Error getting mapper instance. Cause: " + e, e);
  19. }
  20. }

(4)MapperProxyFactory是个苦B的人,粗活最终交给它去做了。咱们看看源码:

  1. /**
  2. * 别人虐我千百遍,我待别人如初恋
  3. * @param mapperProxy
  4. * @return
  5. */
  6. @SuppressWarnings("unchecked")
  7. protected T newInstance(MapperProxy<T> mapperProxy) {
  8. //动态代理我们写的dao接口
  9. return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
  10. }
  11.  
  12. public T newInstance(SqlSession sqlSession) {
  13. final MapperProxy<T> mapperProxy = new MapperProxy<T>(sqlSession, mapperInterface, methodCache);
  14. return newInstance(mapperProxy);
  15. }

通过以上的动态代理,咱们就可以方便地使用dao接口啦, 就像之前咱们写的demo那样:

  1. UserDao userMapper = sqlSession.getMapper(UserDao.class);
  2. User insertUser = new User();

这下方便多了吧, 呵呵, 貌似mybatis的源码就这么一回事儿啊。

别急,还没完, 咱们还没看具体是怎么执行sql语句的呢。

3. Excutor:

接下来,咱们才要真正去看sql的执行过程了。

上面,咱们拿到了MapperProxy, 每个MapperProxy对应一个dao接口, 那么咱们在使用的时候,MapperProxy是怎么做的呢? 源码奉上:

MapperProxy:

  1. /**
  2. * MapperProxy在执行时会触发此方法
  3. */
  4. @Override
  5. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  6. if (Object.class.equals(method.getDeclaringClass())) {
  7. try {
  8. return method.invoke(this, args);
  9. } catch (Throwable t) {
  10. throw ExceptionUtil.unwrapThrowable(t);
  11. }
  12. }
  13. final MapperMethod mapperMethod = cachedMapperMethod(method);
  14. //二话不说,主要交给MapperMethod自己去管
  15. return mapperMethod.execute(sqlSession, args);
  16. }

MapperMethod:

  1. /**
  2. * 看着代码不少,不过其实就是先判断CRUD类型,然后根据类型去选择到底执行sqlSession中的哪个方法,绕了一圈,又转回sqlSession了
  3. * @param sqlSession
  4. * @param args
  5. * @return
  6. */
  7. public Object execute(SqlSession sqlSession, Object[] args) {
  8. Object result;
  9. if (SqlCommandType.INSERT == command.getType()) {
  10. Object param = method.convertArgsToSqlCommandParam(args);
  11. result = rowCountResult(sqlSession.insert(command.getName(), param));
  12. } else if (SqlCommandType.UPDATE == command.getType()) {
  13. Object param = method.convertArgsToSqlCommandParam(args);
  14. result = rowCountResult(sqlSession.update(command.getName(), param));
  15. } else if (SqlCommandType.DELETE == command.getType()) {
  16. Object param = method.convertArgsToSqlCommandParam(args);
  17. result = rowCountResult(sqlSession.delete(command.getName(), param));
  18. } else if (SqlCommandType.SELECT == command.getType()) {
  19. if (method.returnsVoid() && method.hasResultHandler()) {
  20. executeWithResultHandler(sqlSession, args);
  21. result = null;
  22. } else if (method.returnsMany()) {
  23. result = executeForMany(sqlSession, args);
  24. } else if (method.returnsMap()) {
  25. result = executeForMap(sqlSession, args);
  26. } else {
  27. Object param = method.convertArgsToSqlCommandParam(args);
  28. result = sqlSession.selectOne(command.getName(), param);
  29. }
  30. } else {
  31. throw new BindingException("Unknown execution method for: " + command.getName());
  32. }
  33. if (result == null && method.getReturnType().isPrimitive() && !method.returnsVoid()) {
  34. throw new BindingException("Mapper method '" + command.getName()
  35. + " attempted to return null from a method with a primitive return type (" + method.getReturnType() + ").");
  36. }
  37. return result;
  38. }

既然又回到SqlSession了, 那么咱们就看看SqlSession的CRUD方法了,为了省事,还是就选择其中的一个方法来做分析吧。这儿,咱们选择了selectList方法:

  1. public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
  2. try {
  3. MappedStatement ms = configuration.getMappedStatement(statement);
  4. //CRUD实际上是交给Excetor去处理, excutor其实也只是穿了个马甲而已,小样,别以为穿个马甲我就不认识你嘞!
  5. return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
  6. } catch (Exception e) {
  7. throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e);
  8. } finally {
  9. ErrorContext.instance().reset();
  10. }
  11. }

然后,通过一层一层的调用,最终会来到doQuery方法, 这儿咱们就随便找个Excutor看看doQuery方法的实现吧,我这儿选择了SimpleExecutor:

  1. public <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException {
  2. Statement stmt = null;
  3. try {
  4. Configuration configuration = ms.getConfiguration();
  5. StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);
  6. stmt = prepareStatement(handler, ms.getStatementLog());
  7. //StatementHandler封装了Statement, 让 StatementHandler 去处理
  8. return handler.<E>query(stmt, resultHandler);
  9. } finally {
  10. closeStatement(stmt);
  11. }
  12. }

接下来,咱们看看StatementHandler 的一个实现类 PreparedStatementHandler(这也是我们最常用的,封装的是PreparedStatement), 看看它使怎么去处理的:

  1. public <E> List<E> query(Statement statement, ResultHandler resultHandler) throws SQLException {
  2. //到此,原形毕露, PreparedStatement, 这个大家都已经滚瓜烂熟了吧
  3. PreparedStatement ps = (PreparedStatement) statement;
  4. ps.execute();
  5. //结果交给了ResultSetHandler 去处理
  6. return resultSetHandler.<E> handleResultSets(ps);
  7. }

到此, 一次sql的执行流程就完了。 我这儿仅抛砖引玉,建议有兴趣的去看看Mybatis3的源码。

好啦,本次就到此结束啦,最近太忙了, 又该忙去啦。

深入浅出Mybatis系列(十)---SQL执行流程分析(源码篇)的更多相关文章

  1. 深入浅出Mybatis系列十-SQL执行流程分析(源码篇)

    注:本文转载自南轲梦 注:博主 Chloneda:个人博客 | 博客园 | Github | Gitee | 知乎 最近太忙了,一直没时间继续更新博客,今天忙里偷闲继续我的Mybatis学习之旅.在前 ...

  2. Hive SQL执行流程分析

    转自 http://www.tuicool.com/articles/qyUzQj 最近在研究Impala,还是先回顾下Hive的SQL执行流程吧. Hive有三种用户接口: cli (Command ...

  3. SpringMVC执行流程及源码分析

    SpringMVC流程及源码分析 前言 ​ 学了一遍SpringMVC以后,想着做一个总结,复习一下.复习写下面的总结的时候才发现,其实自己学的并不彻底.牢固.也没有学全,视频跟书本是要结合起来一起, ...

  4. django Rest Framework----APIView 执行流程 APIView 源码分析

    在django—CBV源码分析中,我们是分析的from django.views import View下的执行流程,这篇博客我们介绍django Rest Framework下的APIView的源码 ...

  5. Django 基于类的视图(CBV)执行流程 CBV 源码分析

    一.CBV(基于类的视图) 视图是可以调用的,它接受请求并返回响应,这不仅仅是一个函数,Django提供了一些可以用作视图的类的例子,这些允许您通过继承或mixin来构建视图并重用代码. 基本示例 D ...

  6. Struts流程分析+源码分析

    1.初始化工作 读取配置---转换器-----读取插件 当struts-config.xml配置文件加载到内存,则会创建两个map:ActionConfigs,FromBeans.这两个map都交由M ...

  7. 深入浅出Mybatis系列(九)---强大的动态SQL

    上篇文章<深入浅出Mybatis系列(八)---mapper映射文件配置之select.resultMap>简单介绍了mybatis的查询,至此,CRUD都已讲完.本文将介绍mybatis ...

  8. 深入浅出Mybatis系列(九)---强大的动态SQL(转载)

    原文出处:http://www.cnblogs.com/dongying/p/4092662.html 上篇文章<深入浅出Mybatis系列(八)---mapper映射文件配置之select.r ...

  9. 深入浅出Mybatis系列九-强大的动态SQL

    注:本文转载自南轲梦 注:博主 Chloneda:个人博客 | 博客园 | Github | Gitee | 知乎 上篇文章<深入浅出Mybatis系列(八)---mapper映射文件配置之se ...

随机推荐

  1. JAVA NIO——Buffer和FileChannel

    Java NIO和IO的主要区别 IO NIO 面向流 面向缓冲 阻塞IO 非阻塞IO 无 选择器 示例: import java.io.FileInputStream; import java.io ...

  2. [转]ping检测网络连接异常

    转载地址:http://blog.csdn.net/feizxiang3/article/details/26672781 一般来说当出现网络无法连接时,习惯性的用ping命令来ping某个ip地址, ...

  3. java比较版本号

    java比较版本号,比如1.0.3和1.2.1相比较考虑到可以用String的compareTo()方法,代码如下: public class MainClass { public static vo ...

  4. #听云博客大赛#如何在自己的App嵌入听云产品监控App性能

    近日浏览园子文章的时候,发现博客园与听云正在举办“听云原创博文”大赛.最近手上正好正在开发一款iOS的应用,所以就用听云App来监测一下我的App各个指标,为我的应用保驾护航.下面,我就从头到尾演示下 ...

  5. android dialog

    /** * @Title MenuTest.java * @package com.example.standardview * @since * @version 1.0.0 * @author V ...

  6. Python之路,Day1 - Python基础1

    本节内容 Python介绍 发展史 Python 2 or 3? 安装 Hello World程序 变量 用户输入 模块初识 .pyc是个什么鬼? 数据类型初识 数据运算 表达式if ...else语 ...

  7. 首师大附中科创教育平台 我的刷题记录 0304 50095106扔核弹(XDC,你懂的)

    今天给大家献上"C"级题:50095106扔核弹(XDC,你懂的)!! 试题编号:0304   50095106扔核弹(XDC,你懂的) 难度级别:C: 运行时间限制:1000ms ...

  8. 。Java中的一些小细节

    1.main方法. ------任何一个Java程序都有一个main方法,它是程序的入口. ------当执行  “ java + 类名 “  这个命令时,JVM就会去加载这个类,并且寻找这个类中的m ...

  9. 我的git学习

    当遇到不想commit的,而status已经现实出来了,可以使用 git rm -r --cached "fine name or 文件夹" 出现   Git – fatal: U ...

  10. FlexSlider jQuery滑动切换插件 参数

    demo:http://www.sucaihuo.com/jquery/0/6/demo/ FlexSlider是一个非常出色的jQuery滑动切换插件,它支持所有主流浏览器,并有淡入淡出效果.适合所 ...