Mybatis3源码分析(三):解析mapper的xml配置文件

   这一章我们来看看mybatis是怎么读取mapper的xml配置文件并解析其中的sql语句。

    我们还记得是这样配置sqlSessionFactory的:

[java]  view plain copy
  1. "sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">      
  2.    "dataSource" ref="dataSource" />    
  3.    "configLocation" value="classpath:configuration.xml">     
  4.    "mapperLocations" value="classpath:com/xxx/mybatis/mapper/*.xml"/>      
  5.    "typeAliasesPackage" value="com.tiantian.mybatis.model" />      
  6.     

    这里配置了一个mapperLocations属性,它是一个表达式,sqlSessionFactory会根据这个表达式读取包com.xxx.mybaits.mapper下面的所有xml格式文件,那么具体是怎么根据这个属性来读取配置文件的呢?

    答案就在SqlSessionFactoryBean类中的buildSqlSessionFactory方法中:

[java]  view plain copy
  1. if (!isEmpty(this.mapperLocations)) {  
  2.       for (Resource mapperLocation : this.mapperLocations) {  
  3.         if (mapperLocation == null) {  
  4.           continue;  
  5.         }  
  6.   
  7.         try {  
  8.           XMLMapperBuilder xmlMapperBuilder = new XMLMapperBuilder(mapperLocation.getInputStream(),  
  9.               configuration, mapperLocation.toString(), configuration.getSqlFragments());  
  10.           xmlMapperBuilder.parse();  
  11.         } catch (Exception e) {  
  12.           throw new NestedIOException("Failed to parse mapping resource: '" + mapperLocation + "'", e);  
  13.         } finally {  
  14.           ErrorContext.instance().reset();  
  15.         }  
  16.   
  17.         if (logger.isDebugEnabled()) {  
  18.           logger.debug("Parsed mapper file: '" + mapperLocation + "'");  
  19.         }  
  20.       }  
  21.     }  

    mybatis使用XMLMapperBuilder类的实例来解析mapper配置文件。

[java]  view plain copy
  1. public XMLMapperBuilder(Reader reader, Configuration configuration, String resource, Map sqlFragments) {  
  2.     this(new XPathParser(reader, true, configuration.getVariables(), new XMLMapperEntityResolver()),  
  3.         configuration, resource, sqlFragments);  
  4.   }  

  

[java]  view plain copy
  1. private XMLMapperBuilder(XPathParser parser, Configuration configuration, String resource, Map sqlFragments) {  
  2.     super(configuration);  
  3.     this.builderAssistant = new MapperBuilderAssistant(configuration, resource);  
  4.     this.parser = parser;  
  5.     this.sqlFragments = sqlFragments;  
  6.     this.resource = resource;  
  7.   }  

    接着系统调用xmlMapperBuilder的parse方法解析mapper。

[java]  view plain copy
  1. public void parse() {  
  2.     //如果configuration对象还没加载xml配置文件(避免重复加载,实际上是确认是否解析了mapper节点的属性及内容,  
  3.     //为解析它的子节点如cache、sql、select、resultMap、parameterMap等做准备),  
  4.     //则从输入流中解析mapper节点,然后再将resource的状态置为已加载  
  5.     if (!configuration.isResourceLoaded(resource)) {  
  6.       configurationElement(parser.evalNode("/mapper"));  
  7.       configuration.addLoadedResource(resource);  
  8.       bindMapperForNamespace();  
  9.     }  
  10.     //解析在configurationElement函数中处理resultMap时其extends属性指向的父对象还没被处理的节点  
  11.     parsePendingResultMaps();  
  12.     //解析在configurationElement函数中处理cache-ref时其指向的对象不存在的节点(如果cache-ref先于其指向的cache节点加载就会出现这种情况)  
  13.     parsePendingChacheRefs();  
  14.     //同上,如果cache没加载的话处理statement时也会抛出异常  
  15.     parsePendingStatements();  
  16.   }  

     mybatis解析mapper的xml文件的过程已经很明显了,接下来我们看看它是怎么解析mapper的:

[java]  view plain copy
  1. private void configurationElement(XNode context) {  
  2.     try {  
  3.       //获取mapper节点的namespace属性  
  4.       String namespace = context.getStringAttribute("namespace");  
  5.       if (namespace.equals("")) {  
  6.         throw new BuilderException("Mapper's namespace cannot be empty");  
  7.       }  
  8.       //设置当前namespace  
  9.       builderAssistant.setCurrentNamespace(namespace);  
  10.       //解析mapper的节点  
  11.       cacheRefElement(context.evalNode("cache-ref"));  
  12.       //解析mapper的节点  
  13.       cacheElement(context.evalNode("cache"));  
  14.       //解析mapper的节点  
  15.       parameterMapElement(context.evalNodes("/mapper/parameterMap"));  
  16.       //解析mapper的节点  
  17.       resultMapElements(context.evalNodes("/mapper/resultMap"));  
  18.       //解析mapper的节点  
  19.       sqlElement(context.evalNodes("/mapper/sql"));  
  20.       //使用XMLStatementBuilder的对象解析mapper的节点  
  21.     boolean isSelect = sqlCommandType == SqlCommandType.SELECT;  
  22.     //flushCache属性  
  23.     boolean flushCache = context.getBooleanAttribute("flushCache", !isSelect);  
  24.     //useCache属性  
  25.     boolean useCache = context.getBooleanAttribute("useCache", isSelect);  
  26.     //resultOrdered属性  
  27.     boolean resultOrdered = context.getBooleanAttribute("resultOrdered"false);  
  28.   
  29.     // Include Fragments before parsing  
  30.     XMLIncludeTransformer includeParser = new XMLIncludeTransformer(configuration, builderAssistant);  
  31.     includeParser.applyIncludes(context.getNode());  
  32.   
  33.     // Parse selectKey after includes and remove them.  
  34.     processSelectKeyNodes(id, parameterTypeClass, langDriver);  
  35.       
  36.     // Parse the SQL (pre:  and  were parsed and removed)  
  37.     SqlSource sqlSource = langDriver.createSqlSource(configuration, context, parameterTypeClass);  
  38.     //resultSets属性  
  39.     String resultSets = context.getStringAttribute("resultSets");  
  40.     //keyProperty属性  
  41.     String keyProperty = context.getStringAttribute("keyProperty");  
  42.     //keyColumn属性  
  43.     String keyColumn = context.getStringAttribute("keyColumn");  
  44.     KeyGenerator keyGenerator;  
  45.     String keyStatementId = id + SelectKeyGenerator.SELECT_KEY_SUFFIX;  
  46.     keyStatementId = builderAssistant.applyCurrentNamespace(keyStatementId, true);  
  47.     if (configuration.hasKeyGenerator(keyStatementId)) {  
  48.       keyGenerator = configuration.getKeyGenerator(keyStatementId);  
  49.     } else {  
  50.       //useGeneratedKeys属性  
  51.       keyGenerator = context.getBooleanAttribute("useGeneratedKeys",  
  52.           configuration.isUseGeneratedKeys() && SqlCommandType.INSERT.equals(sqlCommandType))  
  53.           ? new Jdbc3KeyGenerator() : new NoKeyGenerator();  
  54.     }  
  55.   
  56.     builderAssistant.addMappedStatement(id, sqlSource, statementType, sqlCommandType,  
  57.         fetchSize, timeout, parameterMap, parameterTypeClass, resultMap, resultTypeClass,  
  58.         resultSetTypeEnum, flushCache, useCache, resultOrdered,   
  59.         keyGenerator, keyProperty, keyColumn, databaseId, langDriver, resultSets);  
  60.   }  
     由以上代码可以看出mybaits使用XPath解析mapper的配置文件后将其中的resultMap、parameterMap、cache、statement等节点使用关联的builder创建并将得到的对象关联到configuration对象中,而这个configuration对象可以从sqlSession中获取的,这就解释了我们在使用sqlSession对数据库进行操作时mybaits怎么获取到mapper并执行其中的sql语句的问题。

你可能感兴趣的:(Java,EE)