1

mybatis源码分析-配置文件解析过程 一文中,主要对配置文件解析过程做了讲解,并且针对插件的配置文件做了详细讲解,本文主要对 mapper 映射文件的解析进行讲解。

mapper文件的加载

首先我们看下mybatis官网,看下mapper文件如何被全局配置文件加载的。下面是官网的说明:


既然 MyBatis 的行为已经由上述元素配置完了,我们现在就要定义 SQL 映射语句了。 但是首先我们需要告诉 MyBatis 到哪里去找到这些语句。 Java 在自动查找这方面没有提供一个很好的方法,所以最佳的方式是告诉 MyBatis 到哪里去找映射文件。 你可以使用相对于类路径的资源引用, 或完全限定资源定位符(包括file:///的 URL),或类名和包名等。例如:

<!-- 使用相对于类路径的资源引用 -->  
<mappers>  
    <mapper resource="org/mybatis/builder/AuthorMapper.xml"/>  
    <mapper resource="org/mybatis/builder/BlogMapper.xml"/> 
    <mapper resource="org/mybatis/builder/PostMapper.xml"/> 
</mappers>
<!-- 使用完全限定资源定位符(URL) -->
<mappers>
    <mapper url="file:///var/mappers/AuthorMapper.xml"/>
    <mapper url="file:///var/mappers/BlogMapper.xml"/>
    <mapper url="file:///var/mappers/PostMapper.xml"/>
</mappers>
<!-- 使用映射器接口实现类的完全限定类名 -->
<mappers>
    <mapper class="org.mybatis.builder.AuthorMapper"/>
    <mapper class="org.mybatis.builder.BlogMapper"/>
    <mapper class="org.mybatis.builder.PostMapper"/>
</mappers>
<!-- 将包内的映射器接口实现全部注册为映射器 -->
<mappers>
    <package name="org.mybatis.builder"/>
</mappers>

上面是mybatis所支持的mapper文件的编写方式,我们下面从源码角度研究一下。

mapper文件解析源码

首先了解一点,我们需要把接口全部加载到 MapperRegistry 类中,使用map保存。

private final Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<>();

这个 MapperRegistry 类暂时不讲,以后会详细讲解。只要知道 Configuration 类有 addMapper 方法可以把接口加入到上面的 map 中即可。下面看具体源码:

private void mapperElement(XNode parent) throws Exception {  
    if (parent != null) {  
        for (XNode child : parent.getChildren()) {  
            if ("package".equals(child.getName())) {  //①
                String mapperPackage = child.getStringAttribute("name");  
                configuration.addMappers(mapperPackage);  
            } else {  
                String resource = child.getStringAttribute("resource");  
                String url = child.getStringAttribute("url");  
                String mapperClass = child.getStringAttribute("class");  
                if (resource != null && url == null && mapperClass == null) {  //②
                    ErrorContext.instance().resource(resource);  
                    InputStream inputStream = Resources.getResourceAsStream(resource);  
                    XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, resource, configuration.getSqlFragments());  
                    mapperParser.parse();  
                } else if (resource == null && url != null && mapperClass == null) { //③ 
                    ErrorContext.instance().resource(url);  
                    InputStream inputStream = Resources.getUrlAsStream(url);  
                    XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, url, configuration.getSqlFragments());  
                    mapperParser.parse();  
                } else if (resource == null && url == null && mapperClass != null) {// ④  
                    Class<?> mapperInterface = Resources.classForName(mapperClass);  
                    configuration.addMapper(mapperInterface);  
                } else {  
                    throw new BuilderException("A mapper element may only specify a url, resource or class, but not more than one.");  
                }  
            }  
        }  
    }  
}
  • ①如果配置文件使用的是使用 package 方式,需要把该包下面的所有接口都存入 map 中。
  • ②如果配置是resource方式,则获取到 mapper 映射文件的路径,然后解析这些文件,注意解析生成的是 XMLMapperBuilder 对象,而不是 XMLConfigBuilder 对象。XMLMapperBuilder 对象的parse方法就可以进行将mapper文件进行解析。
  • ③如果配置是url方式,同 resource。
  • ④如果配置的是 class ,那么直接存入 map ,无需解析。这种方式使用 mapper 接口替代 mapper映射文件的配置方式。

上面不管使用什么方式,都需要在加载全局配置文件的时候,将我们所定义的接口先缓存起来,以便后面接口的调用。但是上面4种配置方式大致可以分为 2 种,一种直接找 mapper xml 配置文件,一种是找接口或者包(包其实下面也是接口)。

本节暂时不讲解那么多,后面章节主要讲解这两种方式。不过可以首先透漏下:

  • 对于 接口或者包 的形式,主要讲解的代码是

image.png

  • 对于其它两种方式
    主要是 XMLMapperBuilder 的研究。

指尖改变世界
27 声望6 粉丝