面试难点:Mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的?
点击关注公众号,Java干货及时送达
作者:张维鹏
一、解析XML:
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:
MappedStatement对象,这里面有两个属性很重要:id:全限定类名+方法名组成的ID sqlSource:当前SQL标签对应的SqlSource对象
MappedStatement对象会被添加到Configuration中,Configuration对象就是Mybatis中的大管家,基本所有的配置信息都维护在这里。当把所有的XML都解析完成之后,Configuration就包含了所有的SQL信息。
全限定类名+方法名” 找到MappedStatement对象,然后解析里面的SQL内容并进行执行即可。二、Dao接口代理:
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.viewscenes.netsupervisor.dao" />
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>@MapperScan("com.xxx.dao"),beanClass设置为MapperFactoryBean,MapperFactoryBean实现了FactoryBean接口,俗称工厂Bean。那么,当我们通过@Autowired注入这个Dao接口时,返回的对象就是MapperFactoryBean这个工厂Bean中的getObject()方法对象。MapperProxy,当我们通过@Autowired注入Dao接口时,注入的就是这个代理对象,我们调用 Dao接口中的方法时,则会调用到MapperProxy对象的invoke()方法。三、执行:
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并返回。
四、总结
SqlSource以及动态标签SqlNodeMappedStatement对象Spring工厂Bean以及动态代理 SqlSession以及执行器
namespace+id唯一即可。
往 期 推 荐
4、为什么国外JetBrains做 IDE 就可以养活自己,国内不行?区别在哪?
点分享
点收藏
点点赞
点在看
评论





