MyBatis實(shí)現(xiàn)配置加載的步驟
前言
在原始MyBatis的使用中,使用MyBatis時(shí)會(huì)先讀取配置文件mybatis-config.xml為字符流或者字節(jié)流,然后通過SqlSessionFactoryBuilder基于配置文件的字符流或字節(jié)流來構(gòu)建SqlSessionFactory。
本篇文章將結(jié)合MyBatis源碼,對(duì)讀取配置文件mybatis-config.xml和構(gòu)建SqlSessionFactory的原理進(jìn)行學(xué)習(xí)。
正文
原始MyBatis讀取配置文件mybatis-config.xml和構(gòu)建SqlSessionFactory的一個(gè)示例如下。
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);上述示例中的Resources工具類提供了方法可以讀取classpath下指定名字的文件為字符流或者字節(jié)流,這里是使用了其提供的getResourceAsStream() 方法將mybatis-config.xml 文件讀取為字節(jié)流。SqlSessionFactoryBuilder是一個(gè)建造者,其提供了共計(jì)9個(gè)重載的build() 方法用于構(gòu)建SqlSessionFactory,這9個(gè)build() 方法可以分為三類,概括如下。
- 基于配置文件字符流構(gòu)建SqlSessionFactory;
- 基于配置文件字節(jié)流構(gòu)建SqlSessionFactory;
- 基于Configuration類構(gòu)建SqlSessionFactory。
實(shí)際上,基于配置文件字符流和基于配置文件字節(jié)流構(gòu)建的方式,最終都是將字符流或字節(jié)流解析并生成Configuration類,然后基于Configuration類構(gòu)建SqlSessionFactory。
下面以基于配置文件字節(jié)流構(gòu)建SqlSessionFactory的過程為例,對(duì)整個(gè)讀配置文件的流程進(jìn)行說明。上面的示例中調(diào)用的build() 方法如下所示。
public SqlSessionFactory build(InputStream inputStream) {
return build(inputStream, null, null);
}上面被調(diào)用的重載的build() 方法如下所示。
public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {
try {
// XMLConfigBuilder會(huì)解析配置文件并生成Configuration類
XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);
// 調(diào)用入?yún)镃onfiguration的build()方法構(gòu)建SqlSessionFactory并返回
return build(parser.parse());
} catch (Exception e) {
throw ExceptionFactory.wrapException("Error building SqlSession.", e);
} finally {
ErrorContext.instance().reset();
try {
inputStream.close();
} catch (IOException e) {
}
}
}可以發(fā)現(xiàn),配置文件的解析是發(fā)生在XMLConfigBuilder的parse() 方法中,在查看parse() 方法前,先看一下XMLConfigBuilder的類圖,如下所示。

通過XMLConfigBuilder的類圖可以知道,XMLConfigBuilder解析配置文件是依靠XPathParser,而XPathParser是MyBatis提供的基于JAVA XPath的解析器。同時(shí),XMLConfigBuilder內(nèi)部維護(hù)了一個(gè)Configuration,通過XPathParser解析配置文件得到的配置屬性均會(huì)豐富到Configuration中。
現(xiàn)在開始分析XMLConfigBuilder的parse() 方法,其實(shí)現(xiàn)如下所示。
public Configuration parse() {
if (parsed) {
throw new BuilderException("Each XMLConfigBuilder can only be used once.");
}
parsed = true;
parseConfiguration(parser.evalNode("/configuration"));
return configuration;
}要理解parse() 方法,最好是和一個(gè)實(shí)際的配置文件進(jìn)行對(duì)照,如下給出一個(gè)實(shí)際的配置文件。
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<settings>
<setting name="useGeneratedKeys" value="true"/>
</settings>
<environments default="development">
<environment id="development">
<transactionManager type="JDBC"/>
<dataSource type="POOLED">
<property name="driver" value="com.mysql.cj.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/test?characterEncoding=utf-8&serverTimezone=UTC&useSSL=false"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</dataSource>
</environment>
</environments>
<mappers>
<package name="com.mybatis.learn.dao"/>
</mappers>
</configuration>所以在parse() 方法中,首先是獲取配置文件的configuration節(jié)點(diǎn)(根節(jié)點(diǎn)),然后將configuration節(jié)點(diǎn)傳入parseConfiguration() 方法,接著在parseConfiguration() 方法中會(huì)根據(jù)傳入的configuration節(jié)點(diǎn)依次獲取子節(jié)點(diǎn)并讀取子節(jié)點(diǎn)屬性,最后將獲取到的屬性豐富進(jìn)Configuration。parseConfiguration() 方法實(shí)現(xiàn)如下所示。
private void parseConfiguration(XNode root) {
try {
propertiesElement(root.evalNode("properties"));
Properties settings = settingsAsProperties(root.evalNode("settings"));
loadCustomVfs(settings);
loadCustomLogImpl(settings);
typeAliasesElement(root.evalNode("typeAliases"));
pluginElement(root.evalNode("plugins"));
objectFactoryElement(root.evalNode("objectFactory"));
objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));
reflectorFactoryElement(root.evalNode("reflectorFactory"));
settingsElement(settings);
// 豐富environments標(biāo)簽及其子標(biāo)簽的屬性到Configuration中
environmentsElement(root.evalNode("environments"));
databaseIdProviderElement(root.evalNode("databaseIdProvider"));
typeHandlerElement(root.evalNode("typeHandlers"));
// 豐富mappers標(biāo)簽的屬性到Configuration中
mapperElement(root.evalNode("mappers"));
} catch (Exception e) {
throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);
}
}最后parse() 方法會(huì)返回Configuration,SqlSessionFactoryBuilder會(huì)基于Configuration創(chuàng)建DefaultSqlSessionFactory并返回,如下所示。
public SqlSessionFactory build(Configuration config) {
return new DefaultSqlSessionFactory(config);
}DefaultSqlSessionFactory類圖如下所示。

至此,讀取配置文件mybatis-config.xml和構(gòu)建SqlSessionFactory的基本原理已經(jīng)介紹完畢。
總結(jié)
本篇文章對(duì)MyBatis讀取配置文件并構(gòu)建SqlSessionFactory的一個(gè)整體流程進(jìn)行了介紹,可以概括如下。
- Resources工具類獲取配置文件輸入字符流或字節(jié)流;
- 創(chuàng)建SqlSessionFactoryBuilder;
- SqlSessionFactoryBuilder中會(huì)創(chuàng)建XMLConfigBuilder來解析配置文件得到全局唯一的Configuration;
- 基于Configuration創(chuàng)建DefaultSqlSessionFactory。
實(shí)際上,關(guān)于MyBatis配置文件的讀取,最關(guān)鍵的部分在于如何注冊(cè)映射文件/映射接口,該部分內(nèi)容,會(huì)在后面的文章中進(jìn)行學(xué)習(xí)。
到此這篇關(guān)于MyBatis實(shí)現(xiàn)配置加載的步驟的文章就介紹到這了,更多相關(guān)MyBatis 配置加載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot整合Ehcache3的實(shí)現(xiàn)步驟
本文主要介紹了SpringBoot整合Ehcache3的實(shí)現(xiàn)步驟,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
Spring?MVC中@Controller和@RequestMapping注解使用
這篇文章主要介紹了Spring?MVC中@Controller和@RequestMapping注解使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
利用EasyExcel導(dǎo)出帶有選擇校驗(yàn)框的excel
EasyExcel是一個(gè)輕量級(jí)的Excel處理工具,支持Excel?2003(xls)和Excel?2007及以上版本(xlsx)的文件格式,本文將利用EasyExcel導(dǎo)出帶有選擇校驗(yàn)框的excel,需要的可以參考下2024-12-12
Java實(shí)現(xiàn)AOP面向切面編程的實(shí)例教程
這篇文章主要介紹了Java實(shí)現(xiàn)AOP面向切面編程的實(shí)例教程,通常Java中的AOP都是利用Spring框架中造好的輪子來開發(fā),而本文則關(guān)注于Java本身AOP的設(shè)計(jì)模式實(shí)現(xiàn),需要的朋友可以參考下2016-04-04
Mybatis-Plus根據(jù)時(shí)間段去查詢數(shù)據(jù)的實(shí)現(xiàn)示例
這篇文章主要介紹了Mybatis-Plus根據(jù)時(shí)間段去查詢數(shù)據(jù)的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
springboot整合JPA訪問Mysql的實(shí)現(xiàn)方法
本文主要介紹了springboot整合JPA訪問Mysql的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02

