前言

在原始Mybatis的应用中,应用Mybatis时会先读取配置文件mybatis-config.xml为字符流或者字节流,而后通过SqlSessionFactoryBuilder基于配置文件的字符流或字节流来构建SqlSessionFactory。本节将联合Mybatis源码,对读取配置文件mybatis-config.xml和构建SqlSessionFactory的原理进行学习。

注释

原始Mybatis读取配置文件mybatis-config.xml和构建SqlSessionFactory的一个示例如下。

InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

上述示例中的Resources工具类提供了办法能够读取classpath下指定名字的文件为字符流或者字节流,这里是应用了其提供的getResourceAsStream()办法将mybatis-config.xml文件读取为字节流。SqlSessionFactoryBuilder是一个建造者,其提供了共计9个重载的build()办法用于构建SqlSessionFactory,这9个build()办法能够分为三类,概括如下。

  • 基于配置文件字符流构建SqlSessionFactory
  • 基于配置文件字节流构建SqlSessionFactory
  • 基于Configuration类构建SqlSessionFactory

实际上,基于配置文件字符流和基于配置文件字节流构建的形式,最终都是将字符流或字节流解析并生成Configuration类,而后基于Configuration类构建SqlSessionFactory

上面以基于配置文件字节流构建SqlSessionFactory的过程为例,对整个读配置文件的流程进行阐明。下面的示例中调用的build()办法如下所示。

public SqlSessionFactory build(InputStream inputStream) {    return build(inputStream, null, null);}

下面被调用的重载的build()办法如下所示。

public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {    try {        //XMLConfigBuilder会解析配置文件并生成Configuration类        XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);        //调用入参为Configuration的build()办法构建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) {        }    }}

能够发现,配置文件的解析是产生在XMLConfigBuilderparse()办法中,在查看parse()办法前,先看一下XMLConfigBuilder的类图,如下所示。

通过XMLConfigBuilder的类图能够晓得,XMLConfigBuilder解析配置文件是依附XPathParser,而XPathParserMybatis提供的基于JAVA XPath的解析器。同时,XMLConfigBuilder外部保护了一个Configuration,通过XPathParser解析配置文件失去的配置属性均会丰盛到Configuration中。

当初开始剖析XMLConfigBuilderparse()办法,其实现如下所示。

public Configuration parse() {    if (parsed) {      throw new BuilderException("Each XMLConfigBuilder can only be used once.");    }    parsed = true;    parseConfiguration(parser.evalNode("/configuration"));    return configuration;}

要了解parse()办法,最好是和一个理论的配置文件进行对照,如下给出一个理论的配置文件。

<?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&amp;serverTimezone=UTC&amp;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节点(根节点),而后将configuration节点传入parseConfiguration()办法,接着在parseConfiguration()办法中会依据传入的configuration节点顺次获取子节点并读取子节点属性,最初将获取到的属性丰盛进ConfigurationparseConfiguration()办法实现如下所示。

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标签及其子标签的属性到Configuration中        environmentsElement(root.evalNode("environments"));        databaseIdProviderElement(root.evalNode("databaseIdProvider"));        typeHandlerElement(root.evalNode("typeHandlers"));        //丰盛mappers标签的属性到Configuration中        mapperElement(root.evalNode("mappers"));    } catch (Exception e) {        throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);    }}

最初parse()办法会返回ConfigurationSqlSessionFactoryBuilder会基于Configuration创立DefaultSqlSessionFactory并返回,如下所示。

public SqlSessionFactory build(Configuration config) {    return new DefaultSqlSessionFactory(config);}

DefaultSqlSessionFactory类图如下所示。

至此,读取配置文件mybatis-config.xml和构建SqlSessionFactory的基本原理曾经介绍结束。

总结

本篇文章是对Mybatis读取配置文件并构建SqlSessionFactory的一个整体流程进行了介绍,即通过Resources工具类获取配置文件输出字符流或字节流,而后通过SqlSessionFactoryBuilder创立DefaultSqlSessionFactory,在SqlSessionFactoryBuilder中是通过XMLConfigBuilder来实现配置文件属性的读取并丰盛进ConfigurationSqlSessionFactoryBuilder也是基于XMLConfigBuilder返回的Configuration创立的DefaultSqlSessionFactory。实际上,对于Mybatis配置文件的读取,最要害的局部在于如何注册映射文件/映射接口,该局部内容,会在前面的文章中进行学习。