MyBatis源码分析(02)Configuration和SqlSessionFactory

2024-06-18 16:58

本文主要是介绍MyBatis源码分析(02)Configuration和SqlSessionFactory,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

分析和学习一个框架的源码是一件非常枯燥和无聊的事(看同事的代码也是如此),所以我们遵循简单的步骤,从粗到细,从我们熟悉的内容到具体内容,一点点来学习mybatis。

其实我们对jdbc是很熟悉的,也了解jdbc的步骤,mybatis的底层是jdbc,那么mybatis在大步骤上一定和jdbc是差不多的,而在每一步上面,mybatis又是如何加入了自己的特色和元素,这是我们要分析和学习的地方。

 

 

 

首先来看一下我们前面写过的代码:

上面的代码第一行和第二行是用来读取资源文件的,最后一行用来打印日志,这两行没有什么问题。从SqlSessionFactory开始,我们来加入几个断点,这样可以查看代码走到了哪里。

我们的第一个起点是:

SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

这个语句调用的是build方法,进入这个方法:

方法中还没有什么具体操作,然后我们记下第二步代码的过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

然后进入下一个方法:

方法中使用了XMLConfigBuilder类,可以看出我们是在xml中做全局配置的,下面记录下现在第三步的过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

下面的这一行build(parser.parse())是对配置的解析,我们进入,看看如何解析,先进入参数中的parse方法内:

从上面我们看到了我们熟悉的mybatis全局配置的 "/configuration" 节点,也是配置文件的根节点,我们记录下第四步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

倒数第二行的方法parseConfiguration英文意思是解析配置的意思,我们进入看看:

在这个方法中,相信大家能看到很多熟悉的关键字,它们都是mybatis全局配置文件中的标签,这个方法其实就是解析读取mybatis全局配置文件的内容,这个方法的参数是root,我们来看一下root的内容:

其实就是我们配置文件的根目录下面的所有内容。而且不同的节点内容会用不同的方法解析,比如propertiesElement、typeAliasesElement、pluginElement等等,下面记录一下第五步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

这些节点解析中,我们重点看一下mapperElement,看看如何解析mappers节点,首先看一下root.evalNode("mappers")这个方法:

进来后调用的也是一个解析方法,我们记录一下第六步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

然后进入下一个解析方法:

我们来记录一下第七步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

7、org.apache.ibatis.parsing.XPathParser#evalNode(java.lang.Object, java.lang.String)

这个方法下面是evaluate方法:

这个方法是所有的解析xml节点的方法,都会调用这里。过程记录一下第八步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

7、org.apache.ibatis.parsing.XPathParser#evalNode(java.lang.Object, java.lang.String)

8、org.apache.ibatis.parsing.XPathParser#evaluate

上面的xpaht是一个接口,执行了evaluate方法,我们来看一下这个方法的具体实现,在idea中,按住ctrl+alt,然后点击方法名:

这是接口的实现方法,记录一下第九步过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

7、org.apache.ibatis.parsing.XPathParser#evalNode(java.lang.Object, java.lang.String)

8、org.apache.ibatis.parsing.XPathParser#evaluate

9、com.sun.org.apache.xpath.internal.jaxp.XPathImpl#evaluate(java.lang.String, java.lang.Object, javax.xml.namespace.QName)

可以看到这一步解析的方法已经不是mybatis中的类了,它使用的是jdk中的方法来解析xml的节点。这个方法中,前面是一系列的判断,最下面的try语句是执行的过程,我们看最后的return语句调用的方法:

最终的xml内容被解析成了集中基本的固定类型数据。关于jdk解析xml的这几个类,大家有兴趣可以自己学习一下。我们记录一下过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

7、org.apache.ibatis.parsing.XPathParser#evalNode(java.lang.Object, java.lang.String)

8、org.apache.ibatis.parsing.XPathParser#evaluate

9、com.sun.org.apache.xpath.internal.jaxp.XPathImpl#evaluate(java.lang.String, java.lang.Object, javax.xml.namespace.QName)

10、com.sun.org.apache.xpath.internal.jaxp.XPathImpl#getResultAsType

上面10步可以跟踪到xml数据的解析过程,我们回头再来看第5步,org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法,

节点解析完后进入的是mapperElement方法,

这个方法的重点是解析mappers,从代码逻辑上可以看到,轮训mappers节点的子节点,然后判断类型是package还是resource或者其它,根据不同的类型做不同的操作,但是最终都执行了一个 configuration.addMapper(mapperInterface)  ,也就是把mappers的配置加到总的configuration配置中,可以想到其它类型的节点也是这样,比如:

我们来记录一下过程,从第五步开始记:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.builder.xml.XMLConfigBuilder#mapperElement

现在再来看上面的第四步,parse方法返回的就是一个Configuration对象,这个对象就是我们前面提到的核心概念的第一个类:

现在我们对mybatis如何组织这个Configuration对象已经了解了,再来看第三步,

这个方法最终调用下面的build返回一个SqlSessionFactory,这个类就是我们前面提到的核心概念的第二个类:

来记录一下:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.session.SqlSessionFactoryBuilder#build(org.apache.ibatis.session.Configuration)

这样我们的第一步代码也就走完了:

SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

现在我们已经知道mybatis如何根据一个全局的xml配置文件生成一个全局的配置对象,然后再根据这个配置对象返回一个SqlSessionFactory对象了。

 

 

 

 

下面总结一下,如果想看单个节点的解析过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.parsing.XNode#evalNode

7、org.apache.ibatis.parsing.XPathParser#evalNode(java.lang.Object, java.lang.String)

8、org.apache.ibatis.parsing.XPathParser#evaluate

9、com.sun.org.apache.xpath.internal.jaxp.XPathImpl#evaluate(java.lang.String, java.lang.Object, javax.xml.namespace.QName)

10、com.sun.org.apache.xpath.internal.jaxp.XPathImpl#getResultAsType

 

 

 

如果想看mybatis全局配置中单个元素的解析过程(例如mappers),也就是全局Configuration对象的生成过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parse方法

5、org.apache.ibatis.builder.xml.XMLConfigBuilder 类的 parseConfiguration 方法

6、org.apache.ibatis.builder.xml.XMLConfigBuilder#mapperElement

 

 

 

如果想看SqlSessionFactory生成的大致过程:

1、SqlSessionFactory sqlSessionFactory =new SqlSessionFactoryBuilder().build(inputStream);

2、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream)方法

3、org.apache.ibatis.session.SqlSessionFactoryBuilder 类的 build(java.io.InputStream, java.lang.String, java.util.Properties)方法

4、org.apache.ibatis.session.SqlSessionFactoryBuilder#build(org.apache.ibatis.session.Configuration)

 

 

使用debug查看源码是一个不错的方法,不过要注意做好记录,否则中间可以会凌乱。从上面可以知道,mybatis的全局配置xml文件最终会组合成一个全局配置对象Configuration,我们进去看一下这个类:

这个类的所有属性和方法都是针对全局配置服务的,我们随便进入一个属性,比如第一个:

这里面就有针对环境配置的所有参数:

mybatis根据这个全局配置对象,生成一个SqlSessionFactory示例,SqlSessionFactory是一个接口,默认使用的实现类是DefaultSqlSessionFactory,可以进去简单看一下:

这里面大部分都是方法,只有一个属性,就是Configuration属性。

 

 

 

 

再对比jdbc,大家可以想一下,加载驱动,使用地址和用户名密码连接数据库,jdbc中只要两行代码就可以,mybatis为了让框架好用又具有适应性,做了很多工作。很多时候看框架源代码也能对我们开发自己的项目有很多启发。

这篇关于MyBatis源码分析(02)Configuration和SqlSessionFactory的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072582

相关文章

将Mybatis升级为Mybatis-Plus的详细过程

《将Mybatis升级为Mybatis-Plus的详细过程》本文详细介绍了在若依管理系统(v3.8.8)中将MyBatis升级为MyBatis-Plus的过程,旨在提升开发效率,通过本文,开发者可实现... 目录说明流程增加依赖修改配置文件注释掉MyBATisConfig里面的Bean代码生成使用IDEA生

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

Spring Boot结成MyBatis-Plus最全配置指南

《SpringBoot结成MyBatis-Plus最全配置指南》本文主要介绍了SpringBoot结成MyBatis-Plus最全配置指南,包括依赖引入、配置数据源、Mapper扫描、基本CRUD操... 目录前言详细操作一.创建项目并引入相关依赖二.配置数据源信息三.编写相关代码查zsRArly询数据库数

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

Spring Boot 整合 MyBatis 连接数据库及常见问题

《SpringBoot整合MyBatis连接数据库及常见问题》MyBatis是一个优秀的持久层框架,支持定制化SQL、存储过程以及高级映射,下面详细介绍如何在SpringBoot项目中整合My... 目录一、基本配置1. 添加依赖2. 配置数据库连接二、项目结构三、核心组件实现(示例)1. 实体类2. Ma