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


    01解析XML

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

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


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

    c93e743f09e2a81c4c01e4671b3e9b38.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配置文件中,一般会这样配置:
    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接口也有了代理实现,所以就可以执行到它里面的方法了。


    03 执行


    如上所述,当我们调用Dao接口方法的时候,实际调用到代理对象的invoke()方法。在这里,实际上调用的就是SqlSession里面的东西了。
    public class DefaultSqlSession implements SqlSession {
     
      public  List 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并返回。

    e461fa224de75f22055985dbd412bfa7.jpeg

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

    2、针对有两个XML文件和这个Dao建立关系是否会冲突的问题:不管有几个XML和Dao建立关系,只要保证namespace+id唯一即可。
  • 相关阅读:
    【RF预测】基于matlab随机森林算法数据回归预测【含Matlab源码 2047期】
    Logger.error还不知道怎么传参打印?看完这个你就明白了
    Hololens低版本不支持多通道或者单眼左眼显示问题
    Linux权限基础知识
    低代码工具大比拼:哪个最适合你?
    客户心声 | 四川省人社厅杨玉成一行充分肯定桂溪街道劳动保障工作信息化建设平台
    Java之BigDecima容器学习心得
    驱动开发:内核特征码搜索函数封装
    Uniapp零基础开发学习笔记(4) -顶部导航栏titleNView的制作
    【低代码】ivx详解之系统架构
  • 原文地址:https://blog.csdn.net/chenxuyuana/article/details/126724912