一、解析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对象看起来应该是这样的:

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

id

全限定类名+方法名组成的ID。

sqlSource

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


到目前为止,XML就解析完成了。看到上面的图示,聪明如你,也许就大概知道了。当我们执行Mybatis方法的时候,就通过全限定类名+方法名找到MappedStatement对象,然后解析里面的SQL内容,执行即可。
二、Dao接口代理
我们的Dao接口并没有实现类,那么,我们在调用它的时候,它是怎样最终执行到我们的SQL语句的呢?
首先,我们在Spring配置文件中,一般会这样配置:

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"><property name="basePackage" value="com.viewscenes.netsupervisor.dao" /><property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>

或者你的项目是基于SpringBoot的,那么肯定也见过这种:br/>@MapperScan("com.xxx.dao")
它们的作用是一样的。将包路径下的所有类注册到Spring Bean中,并且将它们的beanClass设置为MapperFactoryBean。有意思的是,MapperFactoryBean实现了FactoryBean接口,俗称工厂Bean。那么,当我们通过@Autowired注入这个Dao接口的时候,返回的对象就是MapperFactoryBean这个工厂Bean中的getObject()方法对象。br/>那么,这个方法干了些什么呢?
简单来说,它就是通过JDK动态代理,返回了一个Dao接口的代理对象,这个代理对象的处理器是MapperProxy对象。所有,我们通过@Autowired注入Dao接口的时候,注入的就是这个代理对象,我们调用到Dao接口的方法时,则会调用到MapperProxy对象的invoke方法。
对这块内容不太能理解的朋友,可以先看看Spring中的FactoryBean 和 JDK动态代理相关知识
曾经有个朋友问过这样一个问题:

对于有实现的dao接口,mapper还会用代理么?

答案是肯定,只要你配置了MapperScan,它就会去扫描,然后生成代理。但是,如果你的dao接口有实现类,并且这个实现类也是一个Spring Bean,那就要看你在Autowired的时候,去注入哪一个了。
具体什么意思呢?我们来到一个例子。
如果我们给userDao搞一个实现类,并且把它注册到Spring。

@Component
public class UserDaoImpl implements UserDao{public List<User> getUserList(Map<String,Object> map){return new ArrayList<User>();}
}

然后我们在Service方法中,注入这个userDao。猜猜会发生什么?

@Service
public class UserServiceImpl implements UserService{@AutowiredUserMapper userDao1;public List<User> getUserList(Map<String,Object> map) {return userDao1.getUserList(map);}
}

也许你已经猜到了,是的,它会启动报错。因为在注入的时候,找到了两个UserMapper的实例对象。日志是这样的:
No qualifying bean of type [com.viewscenes.netsupervisor.dao.UserDao] is defined: expected single matching bean but found 2: userDaoImpl,userDao
当然了,也许我们的命名不太规范。其实我们通过名字注入就可以了,像这样:
@Autowired UserMapper userDao; 或者 @Autowired UserMapper userDaoImpl; 再或者给你其中一个Bean加上@Primary注解。

说着说着可能扯远了,我们继续回到Mybatis。那么,目前为止,我们通过Dao接口也有了代理实现,所以就可以执行到它里面的方法了。
三、执行
如上所述,当我们调用Dao接口方法的时候,实际调用到代理对象的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以及动态标签SqlNode
MappedStatement对象
Spring 工厂Bean 以及动态代理
SqlSession以及执行器

那么,针对第二个问题:如果有两个XML文件和这个Dao建立关系,岂不是冲突了?
答案也是显而易见,不管有几个XML和Dao建立关系,只要保证namespace+id唯一即可。

转载于:https://blog.51cto.com/13981400/2391902

阿里面试题:Mybatis中的Dao接口和XML文件里的SQL是如何建立关系的?相关推荐

  1. if mybatis tk 多个_面试题:mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的?...

    前言 这是 mybatis 比较常问到的面试题,我自己在以前的面试过程中被问到了2次,2次都是非常重要的面试环节,因此自己印象很深刻. 这个题目我很早就深入学习了,但是一直没有整理出来,刚好最近一段时 ...

  2. .sql文件_面试题:mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的?

    前言 这是 mybatis 比较常问到的面试题,我自己在以前的面试过程中被问到了2次,2次都是非常重要的面试环节,因此自己印象很深刻.这个题目我很早就深入学习了,但是一直没有整理出来,刚好最近一段时间 ...

  3. Mybatis中 Dao接口和XML文件的SQL如何建立关联

    一.解析XML: 首先,Mybatis在初始化 SqlSessionFactoryBean 的时候,找到 mapperLocations 路径去解析里面所有的XML文件,这里我们重点关注两部分. 1. ...

  4. Spring整合MyBatis原理之Mapper接口和xml文件的解析

    目录 1. 前言 2. 类 `SqlSessionFactoryBean` 2.1. 实现了 `FactoryBean` 接口的 `getObject()` 2.2. `buildSqlSession ...

  5. python中利用lxml模块解析xml文件报错XMLSyntaxError: Opening and ending tag mismatch

    今天在代码中第一次使用lxml解析xml文件时出错了, XMLSyntaxError: Opening and ending tag mismatch: keyEffectiveDate line 2 ...

  6. java使用SAX接口解析xml文件

    全栈工程师开发手册 (作者:栾鹏) java教程全解 java使用SAX接口解析xml文件.实现解析类,从xml解析开始到结尾,每一步均会触发对象的重载函数. 测试代码,将book1.xml改成你自己 ...

  7. java给xml加入值,怎么用java读取XML文件里的值并加入jsp页面中的下拉列表中

    如何用java读取XML文件里的值并加入jsp页面中的下拉列表中? 如何用java读取XML文件里的值并加入jsp页面中的下拉列表中? java xml jsp 分享到: ------解决方案---- ...

  8. (转)MyBatis框架的学习(四)——Mapper.xml文件中的输入和输出映射以及动态sql

    http://blog.csdn.net/yerenyuan_pku/article/details/71893689 前面对MyBatis框架的学习中,我们对Mapper.xml映射文件多少有些了解 ...

  9. Mybatis中的核心配置文件SqlMapConfig.xml详细介绍

    一.properties(属性) 可以引用java属性文件中的配置信息如下 jdbc.properties代码如下: jdbc.driver=com.mysql.jdbc.Driver jdbc.ur ...

最新文章

  1. Orleans初战(用分布式解决高并发购物场景)
  2. Webstorm中提示Cannot find module 'webpack.dev.conf.js'
  3. QT使用xsl将xml为html,使用xslt 2.0将属性设置为根元素(Puting attributes to the root-element with xslt 2.0)...
  4. ubuntu 安装 TensorFlow、opencv3 的 tips
  5. 很简单很简单的DBHelper类
  6. layui列表筛选列_Shopify搜索产品并筛选产品列表功能介绍
  7. 修改ssh端口centos7
  8. python中with的用法简单来说_Python中with的用法
  9. php fsockopen,php fsockopen()有什么用?
  10. MIF/MID数据格式说明
  11. html如何解决412问题,网站412怎么解决?
  12. 基于Stata reshape命令的面板数据整理
  13. WindowsXP的图标结构
  14. 你最喜欢的奥斯卡电影是哪部?
  15. 编译安装LNMP及报错解决
  16. MATLAB 使用GUI设计简单的计算器
  17. 痞子衡嵌入式:其实i.MXRT1050,1020,1015系列ROM也提供了FlexSPI driver API
  18. kubernetes endpoint 代码阅读
  19. Python爬虫练习:JS解密,爬取某音漫客网站数据
  20. 跟叶子学把妹——教程序猿把妹第三集

热门文章

  1. Springboot - -web应用开发-Servlets, Filters, listeners
  2. cacti中监控squid的方法
  3. 把现有的typesctipt+react项目接入到electron
  4. python 异常操作 处理方法
  5. eventEmitter3源码分析与学习
  6. UWP crop image control
  7. TopCoder 入门教程 -- sqybi完善版
  8. 黄聪:ICTCLAS分词系统研究(一)(转)
  9. 转:字体集选择font-family
  10. etcd 启动分析_grpc-go基于etcd实现服务发现机制