mybatis 会先解析这些xml 文件,xml 文件里面有命名空间 (namespace),这里可以跟dao 建立关系,然后 xml 中的每段 sql 会有一个id 跟 dao 中的接口进行关联
如果 我有多个xml 文件 都跟这个dao 建立关系了
- 只要保证
namespace+id
唯一即不会冲突 - 具体的关联过程
一解析XML
首先,Mybatis在初始化
SqlSessionFactoryBean
的时候,找到mapperLocations
路径去解析里面所有的XML文件,这里我们重点关注两部分。
创建 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对象应该如下图
![img](file:///C:\Users\19244\AppData\Roaming\Tencent\Users\1924445064\TIM\WinTemp\RichOle\R
F69FB1NNON6MA%~
O`4HO.png)创建MappedStatenemnt
XML文件中的每一个SQL标签就对应一个MappedStatement对象,这里面有两个属性很重要。
- id
全限定类名+方法名组成的ID- sqlSource
当前SQL标签对应的SqlSource对象创建完
MappedStatement
对象,将它缓存到Configuration#mappedStatements
中。
Configuration对象,我们知道它就是Mybatis中的大管家,基本所有的配置信息都维护在这里。把所有的XML都解析完成之后,Configuration就包含了所有的SQL信息。
- 到目前为止,XML就解析完成了。当我们执行Mybatis方法的时候,就通过
全限定类名+方法名
找到MappedStatement
对象,然后解析里面的SQL内容,执行即可。
Configuration(通过全限定类名+方法名)–>MapperStatenment—>SqlSource对象二Dao接口代理
dao接口并没有实现子类,那么我们调用他的时候是怎么执行到xml文件中的SQL语句?
首先 在Spring配置文件中我们一般会这样写
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"> <!--持久层的接口所在的包 --> <property name="basePackage" value="com.contain.dao"></property> <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property> </bean>
来将该路径下所有的类注册到Spring Bean中 将他们的beanClass设置为MapperFactoryBean.
当我们使用@Autowired注入这个dao接口时,返回的对象就是MapperFactoryBean工厂中的getObject()方法对象
- 简单来说,它就是通过JDK动态代理,返回了一个Dao接口的代理对象,这个代理对象的处理器是
MapperProxy
对象。所有,我们通过@Autowired
注入Dao接口的时候,注入的就是这个代理对象,我们调用到Dao接口的方法时,则会调用到MapperProxy
对象的invoke方法。*** 那么问题又来了:对于以实现的Dao接口,mapper还会用代理么 ***
- 答案是肯定的,只要你配置了MapperScan,它就会去扫描,然后生成代理。但是,如果你的dao接口有实现类,并且这个实现类也是一个Spring Bean,它会启动报错。因为在注入的时候,找到了两个UserMapper的实例对象
执行
当我们调用dao接口的时候实际调用的是代理对象的invoke方法,也就是调用SqlSession里面的东西,通过statement
全限定类名+方法名
拿到MappedStatement 对象,然后通过执行器Executor去执行具体SQL并返回。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); } } }
知识点
- SqlSource以及动态SqlNode
- MappedStatement对象
- Spring工厂Bean以及动态代理
- SqlSession以及执行器