mybatis中的dao接口跟.xml文件里面的sql 是如何建立关系的?**

mybatis 会先解析这些xml 文件,xml 文件里面有命名空间 (namespace),这里可以跟dao 建立关系,然后 xml 中的每段 sql 会有一个id 跟 dao 中的接口进行关联

如果 我有多个xml 文件 都跟这个dao 建立关系了

  • 只要保证namespace+id唯一即不会冲突
  • 具体的关联过程

一解析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对象应该如下图
    

    ![img](file:///C:\Users\19244\AppData\Roaming\Tencent\Users\1924445064\TIM\WinTemp\RichOle\RF69FB1NNON6MA%~O`4HO.png)

  2. 创建MappedStatenemnt

XML文件中的每一个SQL标签就对应一个MappedStatement对象,这里面有两个属性很重要。

  • id
    全限定类名+方法名组成的ID
  • sqlSource
    当前SQL标签对应的SqlSource对象

创建完MappedStatement对象,将它缓存到Configuration#mappedStatements中。
Configuration对象,我们知道它就是Mybatis中的大管家,基本所有的配置信息都维护在这里。把所有的XML都解析完成之后,Configuration就包含了所有的SQL信息。

  1. 到目前为止,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以及执行器

你可能感兴趣的:(mybatis,spring,java)