一、解析XML:
首先,Mybatis在初始化 SqlSessionFactoryBean
的时候,找到 mapperLocations
路径去解析里面所有的XML文件,这里我们重点关注两部分。
1、创建SqlSource:
Mybatis会把每个SQL标签封装成SqlSource对象。然后根据SQL语句的不同,又分为动态SQL和静态SQL。其中,静态SQL包含一段String类型的sql语句;而动态SQL则是由一个个SqlNode组成。
假如我们有这样一个SQL:
<select id=”getUserById” resultType=”user”>
select * from user
<where>
<if test=”uid!=null”>
and uid=#{uid}
</if>
</where>
</select>
它对应的SqlSource对象看起来应该是这样的:
2、创建MappedStatement:
ML文件中的每一个SQL标签就对应一个MappedStatement对象,这里面有两个属性很重要。
① id:全限定类名+方法名组成的ID。
② sqlSource:当前SQL标签对应的SqlSource对象。
创建完 MappedStatement
对象,将它缓存到 Configuration 中。Configuration对象就是Mybatis中的大管家,基本所有的配置信息都维护在这里。把所有的XML都解析完成之后,Configuration就包含了所有的SQL信息。
到目前为止,XML就解析完成了。当我们执行Mybatis方法的时候,就通过全限定类名+方法名
找到MappedStatement
对象,然后解析里面的SQL内容,执行即可。
二、Dao接口代理:
我们的Dao接口并没有实现类,那么,我们在调用它的时候,它是怎样最终执行到我们的SQL语句的呢?
首先,我们在Spring配置文件中,一般会这样配置:
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.viewscenes.netsupervisor.dao" />
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>
或者你的项目是基于SpringBoot的,那么肯定也见过这种: @MapperScan("com.xxx.dao")
它们的作用是一样的。将包路径下的所有类注册到Spring Bean中,并且将它们的beanClass设置为 MapperFactoryBean
。MapperFactoryBean
实现了 FactoryBean
接口,俗称工厂Bean。那么,当我们通过 @Autowired
注入这个Dao接口的时候,返回的对象就是MapperFactoryBean
这个工厂Bean中的 getObject()
方法对象。
那么,这个方法干了些什么呢?
简单来说,它就是通过JDK动态代理,返回了一个Dao接口的代理对象,这个代理对象的处理器是MapperProxy
对象。所有,我们通过@Autowired
注入Dao接口的时候,注入的就是这个代理对象,我们调用到Dao接口的方法时,则会调用到MapperProxy
对象的invoke()方法。
那么,目前为止,我们通过Dao接口也有了代理实现,所以就可以执行到它里面的方法了。
三、执行:
如上所述,当我们调用Dao接口方法的时候,实际调用到代理对象的invoke()方法。 在这里,实际上调用的就是SqlSession里面的东西了。
public class DefaultSqlSession implements SqlSession {
public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
try {
MappedStatement ms = configuration.getMappedStatement(statement);
return executor.query(ms,
wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
}
}
}
看到以上代码,说明我们想的不错。它就是通过statement(全限定类型+方法名)
拿到MappedStatement 对象,然后通过执行器Executor去执行具体SQL并返回。
四、总结:
1、针对Mybatis中的Dao接口和XML文件里的SQL是如何建立关系的问题,主要可以归纳为下面几点小点:
- SqlSource以及动态标签SqlNode
- MappedStatement对象
- Spring 工厂Bean 以及动态代理
- SqlSession以及执行器
2、针对有两个XML文件和这个Dao建立关系是否会冲突的问题:不管有几个XML和Dao建立关系,只要保证namespace+id
唯一即可。
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/100036.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...