MyBatis整体预览(一)(一)

2014-11-24 07:43:18 · 作者: · 浏览: 2
题记:最近在工作之余分析了一下MyBatis的源码,促使我阅读源码的原因是为了实现MyBatis在物理上的分页。我们知道,MyBatis是在逻辑上的分页,通过用户的查询,将结果缓存下来,在查看是否传递了RowBounds对象,在查看里面的offset和limit值,通过这两个值,从返回的结果集合中截取位于期间的值。但是这样并不是很好,可以想想,如果假设查询的数据量很大,但是有用的可以是前几条,这未免有点太浪费了。在之前,也在网上查了一下实现分页的方法,最常用的就是添加MyBatis插件,实现Interceptor接口,拦截StatementHandler接口中的prepare方法,后面会介绍为什么拦截这个接口的这个方法。在拦截ResultSetHandler接口的handlerResultSet方法,后面也会对其缘由进行介绍。但是这中方法虽然可以添加分页的SQL语句,但是并没有将分页的offset和limit的值让Mybatis动态的添加到SQL中去,有人会说,可以在拦截StatementHandler接口的时候我们将它们拼装上去。但是这样会容易出现SQL注入的问题。所以这样不得不使我进一步的了解MyBatis的内部原理。本文将就一下几个方面对MyBatis的内部实现进行分析。

\mybatis源码结构图


数据管家——Configuration:
MyBatis在运行期的基本上所有的数据都会汇总到这个类。它的初始数据是来自开发人员配置在configuration的xml配置文件。通过用户配置的environments来获得系统运行的数据库环境,如事物管理以及数据源。下面给出了最基本的配置:
[html]















这些配置对于MyBatis需要做哪些工作呢?通过阅读Configuration的源码会发现,Mybatis其实为configuration标签下面的子标签都有一个对应的变量来进行存储,例如:
[java]
protected final TypeHandlerRegistry typeHandlerRegistry = new TypeHandlerRegistry();

则是存储标签下面配置的所有信息。其他的也类似可以找到。负责创建Configuration对象的则是XMLConfigurationBuilder,这里将完成从配置的XML数据映射到Configuration对象的数据。通过一下方法完成数据的映射:
[java]
private void parseConfiguration(XNode root) {
try {
propertiesElement(root.eva lNode("properties")); typeAliasesElement(root.eva lNode("typeAliases"));
pluginElement(root.eva lNode("plugins"));
objectFactoryElement(root.eva lNode("objectFactory"));
objectWrapperFactoryElement(root.eva lNode("objectWrapperFactory"));
settingsElement(root.eva lNode("settings"));
environmentsElement(root.eva lNode("environments"));
databaseIdProviderElement(root.eva lNode("databaseIdProvider"));
typeHandlerElement(root.eva lNode("typeHandlers"));
mapperElement(root.eva lNode("mappers"));
} catch (Exception e) {
throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);
}
}

可以到,它为每个元素都对应了一个处理方法,这些方法将负责解析我们配置的XML文件。这里面我主要跟踪了几个方法的执行 www.2cto.com
(mapperElement,typeHandlerElement,typeAliasesElement,environmentsElement)
mapperElement——ORM
我们知道,MyBatis支持注解形式和XML形式的ORM配置。那么当然将会有两个类来处理这两种行为,它们分别是XMLMapperBuilder和MapperAnnotationBuilder,它们分别处理什么类型,我看我就不用说了。通过解析configuration/mappers元素来获得ORM配置信息。
1)XML方式的ORM配置和方式,当我们在mappers/mapper的属性中配置了url或者是resource信息的时候将触发MyBatis采用XML的方式进行处理,并读取你指定的mapper路径。在XMLMapperBuilder类中有如下方法:
[java]
private void configurationElement(XNode context) {

try {

String namespace = context.getStringAttribute("namespace");

builderAssistant.setCurrentNamespace(namespace);

cacheRefElement(context.eva lNode("cache-ref"));

cacheElement(context.eva lNode("cache"));

parameterMapElement(context.eva lNodes("/mapper/parameterMap"));

resultMapElements(context.eva lNodes("/mapper/resultMap"));

sqlElement(context.eva lNodes("/mapper/sql"));

buildStatementFromContext(context.eva lNodes("select|insert|update|delete"));

} catch (Exception e) {

throw new RuntimeException("Error parsing Mapper XML. Cause: " + e, e);

}

}

这个方法便是读取你mapper文件中所有制的ORM信息。该方法将通过调用XMLMapperBuilder的parse()方法触发。
2)注解方式配置ORM信息加载,当你配置了mappers/package或者在mapper里