在本系列的第二篇文章中小编就说了,Mybatis的配置信息都是由Configuration来保存的,本篇文章我们就重点来看Mybatis的解析过程。在学习完本篇,你会完全掌握对Mybatis配置的认识,是你产生新的认识。
本篇的源码就从下面的代码片段中开始。
@Test
public void configurationTest() throws Exception {
//拿到mybatis的配置文件输入流
InputStream mapperInputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream("mybatisConfig.xml");
//SqlSessionFactoryBuilder通过XMLConfigBuilder解析器读取配置信息生成Configuration信息
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(mapperInputStream);
//获取配置文件
Configuration configuration = sqlSessionFactory.getConfiguration();
}
SqlSessionFactoryBuilder().build(mapperInputStream);
创建SqlSessionFactoryBuilder.build()
方法中。文档解析器SAX解析,这点不做重点研究,一笔带过。
public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {
try {
XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);
return build(parser.parse());
} catch (Exception e) {
throw ExceptionFactory.wrapException("Error building SqlSession.", e);
} finally {
ErrorContext.instance().reset();
try {
inputStream.close();
} catch (IOException e) {
// Intentionally ignore. Prefer previous error.
}
}
}
XMLConfigBuildery初始化时候已经生成了Document,Mybatis配置文件的具体实现看XMLConfigBuilder.parse()
public Configuration parse() {
if (parsed) {
throw new BuilderException("Each XMLConfigBuilder can only be used once.");
}
parsed = true;
//获取document中需要解析的节点
parseConfiguration(parser.evalNode("/configuration"));
return configuration;
}
private void parseConfiguration(XNode root) {
try {
//根据properties的resource属性去填补配置文件中的占位符
propertiesElement(root.evalNode("properties"));
Properties settings = settingsAsProperties(root.evalNode("settings"));
loadCustomVfs(settings);
//注册Mybatis别名,Mybatis中的引用,都是先从别名注册器中
//或取,当获取不到才直接调动ClassLoader加载。
typeAliasesElement(root.evalNode("typeAliases"));
//解析Mybatis中的插件,Mybatis插件都是拦截器的原理,所以
//获取到拦截器,在执行时候执行拦截器方法即可
pluginElement(root.evalNode("plugins"));
//将sql执行后的数据库返回结果对象,转换指定类型的Java对象///的工厂类。属于MetaObject组件。在DefaultResultSetHandler//使用
objectFactoryElement(root.evalNode("objectFactory"));
objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));
reflectorFactoryElement(root.evalNode("reflectorFactory"));
settingsElement(settings);
//只读取environments节点default属性指定的环境信息
environmentsElement(root.evalNode("environments"));
//Mybatis提供对于不同数据库匹配不同语句的能力,下面讲解。
//获取数据库类型,从Mapper映射文件中读取databaseId对应的数//据库类型配置
databaseIdProviderElement(root.evalNode("databaseIdProvider"));
//类型转换器,这个在数据库表字段,转换Java代码时候很重要
typeHandlerElement(root.evalNode("typeHandlers"));
//mapper的配置文件是一个重要部分,抽出单独说明。
mapperElement(root.evalNode("mappers"));
} catch (Exception e) {
throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);
}
}
parseConfiguration方法才是真正将XML解析成Java代码的地方。我们看Mybatis都如何去解析自己的配置节点
我们以下面的配置文件为例子,看Mybatis如何解析,已经了解他们在Mybatis中的作用
<configuration>
<properties resource="application.properties">properties>
<settings>
<setting name="logImpl" value="LOG4J"/>
settings>
<plugins>
<plugin interceptor="orm.example.interceptor.CustomerInterceptor">
<property name="初始化key" value="初始化value">property>
plugin>
plugins>
<environments default="development">
<environment id="development">
<transactionManager type="JDBC"/>
<dataSource type="POOLED">
<property name="driver" value="${spring.datasource.driver-class-name}"/>
<property name="url" value="${spring.datasource.url}"/>
<property name="username" value="${spring.datasource.username}"/>
<property name="password" value="${spring.datasource.password}"/>
dataSource>
environment>
environments>
<databaseIdProvider type="org.apache.ibatis.mapping.VendorDatabaseIdProvider" >
<property name="MySQL" value="mysql"/>
<property name="Oracle" value="oracle" />
databaseIdProvider>
<mappers>
<mapper resource="mapper/TUserMapper.xml"/>
mappers>
configuration>
propertiesElement(root.evalNode("properties"));
从配置文件中,读取properties
标签,生成Properties对象,交给XPathParser保管,在解析配置节点的属性时候
当遇到${}
站位符,就从properties指定的文件中读取数值。
这点的处理逻辑在PropertyParser
实现,想要深入研究自行学习。
Properties settings = settingsAsProperties(root.evalNode("settings"));
读取settings
节点,生成Properties对象,Mybatis会根据其,去用户化配置系统的实现类,或者是默认值。
这点的逻辑逻辑在settingsElement(Properties props)
方法中,比如: 上面配置文件中自定义了日志的实现类。logImpl=>LOG4J
typeAliasesElement(root.evalNode("typeAliases"));
细心的同学可能发现,settings中logImpl的实现,配置文件中写的是LOG4J。那Mybatis是如何根据LOG4J找到具体的Class对象的呢?
其实就是typeAliases标签来,配置的,但是我们的配置文件中没有写这个节点呀,Mybatis是如何做的呢? 这里我们就要引入一个类
TypeAliasRegistry
。小编叫别名注册器,代码非常简单。
pluginElement(root.evalNode("plugins"));
Mybatis插件都是拦截器的原理,读取拦截器实现类,添加到Configuration
的拦截链中InterceptorChain
,在执行时候执行拦截器的逻辑
configuration.addInterceptor(interceptorInstance);
public void addInterceptor(Interceptor interceptor) {
interceptorChain.addInterceptor(interceptor);
}
这里我们主要分析下Mybatis的拦截器接口,以及如何去扩展它
/**
* @author Clinton Begin
*/
public interface Interceptor {
//里面实现主要实现拦截逻辑,参数是代理类等
Object intercept(Invocation invocation) throws Throwable;
//一般使用Plugin.wrap(target, this);加载当前插件,解析拦截器的声明注解@Intercepts和@Signature, 包装模式,属于结构性设计模式
Object plugin(Object target);
//初始化属性 plugins配置节点的子节点plugin的property
<!--<plugins>-->
<!-- <plugin interceptor="com.github.pagehelper.PageInterceptor">-->
<!-- <property name="初始化key" value="初始化value"></property>-->
<!-- </plugin>-->
<!--</plugins>-->
void setProperties(Properties properties);
}
自定的拦截器仅仅继承Interceptor是不够的,同时也要用@Intercepts和@Signature来修饰,Plugin.wrap(target, this)
时候会解析这两个注解。
那么如何使用这两个注解呢
Type的类型只能是: StatementHandler | ParameterHandler | ResultSetHandler | Executor 类或者子类
@Test
public void CustomerInterceptorTest(){
//拿到mybatis的配置文件输入流
InputStream mapperInputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream("mybatisConfig.xml");
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(mapperInputStream);
TUserMapper tUserMapper = sqlSessionFactory.getConfiguration().getMapper(TUserMapper.class, sqlSessionFactory.openSession());
System.out.println(tUserMapper.selectOne(2));
}
以上就是Mybatis的拦截器,推荐可以看看最经典的拦截器: 分页插件。它的逻辑就是,生成Page分页对象,在执行查询时候拼装sql的limit信息,有兴趣的可以深入研究。
objectFactoryElement(root.evalNode("objectFactory"));
objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));
reflectorFactoryElement(root.evalNode("reflectorFactory"));
ObjectFactory、ObjectWrapperFactory、ReflectorFactory这三个对象都属于MetaObject的组件,从名字上看就是生成对象的,主要用做
数据库返回结果,通过这三个类,生成Java中对应的类。
environmentsElement(root.evalNode("environments"));
比较容易解释了,读取environments节点default属性指定的环境信息。
databaseIdProviderElement(root.evalNode("databaseIdProvider"));
Mybatis是支持多种数据库的,在使用Mybatis编写Mapper的时候,可以为同一个方法编写多种数据库的描述,Mybatis在启动时候会自动读取数据库类型,然后根据类型去查询Mapper配置中databaseId对应的sql信息。
主要Mysql数据库名字是MySQL,Oracle的名字是Oracle,配置文件中的value要与Mapper配置中的databaseId相对应。一般开发很少这样去写。虽然Mybatis提供这样的能力。
typeHandlerElement(root.evalNode("typeHandlers"));
这个也比较容易理解,类的设计很简单和TypeAliasRegistry很类似具体实现类是TypeHandlerRegistry。就是负责数据库类型和Java类型的转换。
mapperElement(root.evalNode("mappers"));
终于读取到Mapper信息了,这个是Mybatis,对象关系映射核心的处理逻辑。
sql的配置信息BoundSql对象,这里面涉及到占位符和变量符的问题。这点在<<深入浅出Mybatis系列(二)Mybatis核心配置篇>>已经讲过了。
而Mapper类在Mybatis最终会转换成方法签名MethodSignature
什么是方法签名,就是对一个方法的描述。比如: 返回值类型是Void,是集合还是实体类。以及获取@Param注解标记的入参的别名。
以及在执行数据库交互时候,获取接口的入参。这部分逻辑在MapperMethod
中,MethodSignature.convertArgsToSqlCommandParam()
方法用来处理@Param
注解。如下图所示。
上面我们其实已经讲了,Mybatis是如何拿到Mapper类的参数了,以及Mapper类在Java中信息保存的载体。那么如何拿到BoundSql呢? 其实就在MappedStatement
中,MappedStatement
即包含了MethodSignature
也包含了BoundSql
。而MappedStatement是在应用启动时候读取mappers
配置节点时候就生成的并保存在Configuration
。
public class Configuration {
protected final MapperRegistry mapperRegistry = new MapperRegistry(this);
protected final InterceptorChain interceptorChain = new InterceptorChain();
protected final TypeHandlerRegistry typeHandlerRegistry = new TypeHandlerRegistry();
protected final TypeAliasRegistry typeAliasRegistry = new TypeAliasRegistry();
protected final LanguageDriverRegistry languageRegistry = new LanguageDriverRegistry();
protected final Map<String, MappedStatement> mappedStatements = new StrictMap<MappedStatement>("Mapped Statements collection");
...
}
MapperBuilderAssistant是具体生成MappedStatement的。而调用MapperBuilderAssistant有两种实现
到这里mappers
节点也已经读取完了,Mapper类方法签名信息保存在MethodSignature,Sql信息保存在BoundSql。在执行sql时候,会将参数与sql拼装起来使用,当拼装好的sql发送到数据库服务器,并受到服务器返回后,Mybatis会将数据库对象,转行成Java类数据对象。这部分逻辑我们下一篇主要来讲解。另外这里涉及到BoundSql的有两个知识点: 1.占位符# 2.变量符$ 在本系列第二篇已经说过了,这里不再赘述。