面试官:Mybatis中 Dao接口和XML文件的SQL如何建立关联?

01解析XML

首先,Mybatis 在初始化 SqlSessionFactoryBean 时,会找到 mapperLocations 配置的路径下中所有的XML文件并进行解析,这里我们重点关注两部分:
1、创建SqlSource:
Mybatis会把每个SQL标签封装成SqlSource对象,然后根据SQL语句的不同,又分为动态SQL和静态SQL。其中,静态SQL包含一段String类型的sql语句;而动态SQL则是由一个个SqlNode组成。
3e254a8999a667c73571f3a6c8059fd7.jpeg
假如我们有这样一个SQL:

<select&nbsp;id="getUserById"&nbsp;resultType="user">
&nbsp;&nbsp;&nbsp;&nbsp;select&nbsp;* from&nbsp;user
&nbsp;&nbsp;&nbsp;&nbsp;<where>
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<if&nbsp;test="uid!=null">
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;and uid=#{uid}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</if>
&nbsp;&nbsp;&nbsp;&nbsp;</where>
</select>


它对应的SqlSource对象看起来应该是这样的:

d5e30d4c8570d3f69912e2e3e5be8bb0.jpeg


2、创建MappedStatement:
接下来,Mybatis会为XML中的每个SQL标签都生成一个MappedStatement对象,这里面有两个属性很重要:
  • ① id:全限定类名+方法名组成的ID
  • ② sqlSource:当前SQL标签对应的SqlSource对象

创建完的 MappedStatement 对象会被添加到 Configuration 中,Configuration对象就是Mybatis中的大管家,基本所有的配置信息都维护在这里。当把所有的XML都解析完成之后,Configuration就包含了所有的SQL信息。

db0c5eb10dd933f595d2410a8022f832.jpeg

到目前为止,XML就解析完成了,当我们执行Mybatis方法的时候,就可以通过 “全限定类名+方法名” 找到 MappedStatement 对象,然后解析里面的SQL内容并进行执行即可。


02 Dao接口代理


但是Dao接口并没有具体的实现类,那么在被调用时,最终又是怎样找到我们的SQL语句的呢?
首先,我们在Spring配置文件中,一般会这样配置:
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property&nbsp;name="basePackage"&nbsp;value="com.viewscenes.netsupervisor.dao"&nbsp;/>
<property&nbsp;name="sqlSessionFactoryBeanName"&nbsp;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接口也有了代理实现,所以就可以执行到它里面的方法了。


03 执行


如上所述,当我们调用Dao接口方法的时候,实际调用到代理对象的invoke()方法。在这里,实际上调用的就是SqlSession里面的东西了。
public&nbsp;class&nbsp;DefaultSqlSession&nbsp;implements&nbsp;SqlSession&nbsp;{
&nbsp;
&nbsp;&nbsp;public&nbsp;<E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;MappedStatement ms = configuration.getMappedStatement(statement);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;executor.query(ms,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;}
}

看到以上代码,说明我们想的不错。它就是通过statement(全限定类型+方法名)拿到MappedStatement 对象,然后通过执行器Executor去执行具体SQL并返回。

e461fa224de75f22055985dbd412bfa7.jpeg

04 总结
1、针对Mybatis中的Dao接口和XML文件里的SQL是如何建立关系的问题,主要可以归纳为下面几点小点:
  • SqlSource以及动态标签SqlNode
  • MappedStatement对象
  • Spring 工厂Bean 以及动态代理
  • SqlSession以及执行器

2、针对有两个XML文件和这个Dao建立关系是否会冲突的问题:不管有几个XML和Dao建立关系,只要保证namespace+id唯一即可。
MyBatis配置mapper接口XML文件关联,通常需要遵循以下步骤: 1. 创建Mapper接口:首先,你需要创建一个普通的Java接口,这个接口定义了各种你需要执行的数据库操作方法。例如,如果你要操作用户表,可以定义一个名为UserMapper的接口。 2. 编写XML映射文件:接着,你需要编写一个与Mapper接口同名的XML文件,通常放在与Mapper接口相同的包路径下。在这个XML文件,你需要配置具体的SQL语句,并通过`<select>`、`<insert>`、`<update>`、`<delete>`等标签来定义各个方法对应的SQL操作。例如,如果UserMapper接口有一个方法是获取用户信息,那么在UserMapper.xml就会有一个`<select>`标签对应的SQL查询语句。 3. 配置文件关联:为了让MyBatis知道mapper接口XML文件之间的关系,你需要在MyBatis的全局配置文件(通常是mybatis-config.xml,使用`<mappers>`标签来注册mapper接口的位置。可以通过`<mapper resource="完整路径/UserMapper.xml"/>`来指定XML文件的位置。 4. 使用SqlSessionFactoryBuilder构建SqlSessionFactory:MyBatis使用SqlSessionFactory来构建SqlSession,而SqlSession负责实际的数据库操作。你需要使用SqlSessionFactoryBuilder来根据MyBatis全局配置文件构建SqlSessionFactory。 5. 通过SqlSession获取Mapper代理:通过SqlSession对象,你可以调用getMapper()方法,并传入Mapper接口的Class对象,从而获取到这个接口的动态代理对象。之后就可以直接调用接口方法,MyBatis会根据提供的信息完成SQL的执行。 示例代码如下: **UserMapper.java** ```java public interface UserMapper { User selectUserById(int id); int insertUser(User user); // 其他方法... } ``` **UserMapper.xml** ```xml <mapper namespace="com.example.mapper.UserMapper"> <select id="selectUserById" resultType="com.example.model.User"> SELECT * FROM users WHERE id = #{id} </select> <insert id="insertUser"> INSERT INTO users(name, age) VALUES (#{name}, #{age}) </insert> <!-- 其他SQL配置... --> </mapper> ``` **mybatis-config.xml** ```xml <configuration> <!-- 其他配置... --> <mappers> <mapper resource="com/example/mapper/UserMapper.xml"/> <!-- 其他mapper配置... --> </mappers> </configuration> ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值