2020-09-20 mybatis学习笔记

  1. '#{}' 和'${}'的区别是什么?
 ${}是Properties文件中的变量占位符,它可以用于标签属性值和sql内部,属于静态文本替换,比如${driver}
会被静态替换为 com.mysql.jdbc.Driver。
 #{}是sql的参数占位符,MyBatis会将sql中的#{}替换为?号,sql执行前会使用PerparedStatement的参数设置
方法,按序给sql的?号占位符设置参数值。比如 #{item.name}的取值方式为使用反射从参数对象中获取item
对象的name属性值,相当于param.getItem().getName()。#{}只能用于sql拼接中的参数,将?的参数值替换为#{value}

 #{} 和 ${} 在预编译中的处理是不一样的。
#{} 在预处理时,会把参数部分用一个占位符 ? 代替, ${} 则只是简单的字符串替换。

在使用过程中我们应该使用哪种方式呢?
答案是:优先使用 #{}。因为 ${} 会导致 sql 注入的问题。

2.XML映射文件的标签

MyBatis常见的标签,除了标签,都会被解析成一个MappedStatement对象。
Dao 接口里的方法可以重载,但是Mybatis的XML里面的ID不允许重复。
/**
 * Mapper接口里面方法重载
 */
public interface StuMapper {

    List getAllStu();
    
    List getAllStu(@Param("id") Integer id);
}

//然后在 StuMapper.xml 中利用Mybatis的动态sql就可以实现。
    

Dao接口的工作原理是JDK动态代理,MyBatis运行时会使用JDK动态代理为Dao接口生成代理proxy对象,代理对象proxy会拦截
接口方法,转而执行Mapped所代表的sql,然后将sql执行结果返回。

5.简述Mybatis的插件运行原理,以及如何编写一个插件

Mybatis仅可以编写针对ParameterHandler、ResultSetHandler、StatementHandler、Executor这四种接口的插件,Mybatis使用
jdk的动态代理,为需要拦截的接口生成代理对象,以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截
方法,具体就是IvocationHandler的invoke()方法。当然,只会拦截哪些你指定需要拦截的方法。
实现MyBatis中的Interceptor接口并复写intercept()方法,然后给插件编写注解,指定要拦截哪一个接口的方法,最后在配置文件
中配置编写的插件。

8.Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?

第一种是使用标签,逐一定义列名和对象属性名之间的映射关系。第二种是使用sql列的别名功能,将列别名书写为
对象属性名。Mybatis会忽略列名大小写,找到与之对应对象属性名。
有了列名与属性名映射关系后,mybatis通过反射创建对象,用反射给对象的属性赋值并返回,找不到映射关系的属性,无法赋值。

9.Mybatis能执行一对一、一对多查询吗?有哪些实现方式,以及它们之间的区别。

能,Mybatis不仅可以执行一对一、一对多关联查询,还可以执行多对一,多对多的关联查询。多对一查询,其实就是一对一查
询,只需要把selectOne()修改为selectList()即可;多对多查询,其实就是一对多查询,只需要把selectOne()修改为selectList()即
可。
关联对象查询有两种实现方式,一种是单独发送一个sql去查询关联对象,赋给主对象,然后返回主对象。另一种是使用嵌套查
询,嵌套查询的含义为使用join查询,一部分A对象的属性值,另一部分是关联对象的属性值,好处是只发一个sql查询,就可以把
主对象和其关联对象查出来。

10.Mybatis是否支持延迟加载?如果支持,它的实现原理是什么?

Mybatis仅支持association关联对象和collection关联对象集合的延迟加载,association指的是一对一,collection指的是一对多查
询。在Mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled = true|false;
它的原理是,使用cglib创建目标对象的代理对象,当调用目标方法时,进入拦截方法,比如调用a.getB().getName(),拦截器
invoke()方法发现a.getB()是null值,那么会单独发送事先保存好的查询关联B对象的sql,把B查询上来,然后调用a.setB(b),于是a
的对象b属性就有值了,接着完成a.getB().getName()。

11、Mybatis都有哪些Executor执行器?它们之间的区别是什么?

Mybatis有三种基本的Executor执行器,SimpleExecutor、ReuseExecutor、BatchExecutor。
- SimpleExecutor:每执行一次update或select,就开启一个statement对象,用完立即关闭Statement对象。
- ReuseExecutor:执行update或select,以sql作为key查找statement对象,存在就使用,不存在就创建,用完后不关闭Statement
对象,而是放置在Map中,供下一次使用。
- BatchExecuotr:执行update(没有select,JDBC批处理不支持select),将所有sql都添加到批处理中(addBatch()),等待统一
执行(executeBatch()),它缓存了多个Statement对象,每个statement对象都是addBatch()完毕后,等待逐一执行executeBatch()批处理。

12.MyBatis如何制定使用哪一种Executor执行器?

在MyBatis配置文件中,可以指定默认的ExecutorType执行器类型,也可以手动给DefaultSqlSessionFactory的创建SqlSession的
方法传递ExecutorType类型参数。

13.mybatis工作原理

(1)读取核心配置文件并返回InputStream流对象。
(2)根据InputStream流对象解析出Configuration对象,然后创建SqlSessionFactory工厂对象
(3)根据一系列属性从SqlSessionFactory工厂中创建SqlSession
(4)从SqlSession中调用Executor执行数据库操作,生成具体sql指令。
(5)对执行结果进行二次封装
(6)提交与事务

14.MyBatis 是否可以映射 Enum 枚举类?

MyBatis 可以映射枚举类,不单可以映射枚举类,MyBatis 可以映射任何对象到表的一列上。
映射方式为自定义一个 `TypeHandler`,实现 `TypeHandler` 的 `setParameter()`和 `getResult()`接口方法。
`TypeHandler` 有两个作用,一是完成从 javaType 至 jdbcType 的转换,二是完成 jdbcType 至 javaType 的转
换,体现为 `setParameter()`和 `getResult()`两个方法,分别代表设置 sql 问号占位符参数和获取列查询结果。

15.MyBatis 映射文件中,如果 A 标签通过 include 引用了 B 标签的内容,请问,B 标签能否定义在 A 标签的后面,还是说必须定义在 A 标签的前面?

虽然 MyBatis 解析 Xml 映射文件是按照顺序解析的,但是,被引用的 B 标签依然可以定义在任何地方,MyBatis 都可以正确识别。

原理是,MyBatis 解析 A 标签,发现 A 标签引用了 B 标签,但是 B 标签尚未解析到,尚不存在,此时,
MyBatis 会将 A 标签标记为未解析状态,然后继续解析余下的标签,包含 B 标签,待所有标签解析完毕,
MyBatis 会重新解析那些被标记为未解析的标签,此时再解析 A 标签时,B 标签已经存在,A 标签也就可以正常解析完成了。

16.简述 MyBatis 的 Xml 映射文件和 MyBatis 内部数据结构之间的映射关系?

MyBatis 将所有 Xml 配置信息都封装到 All-In-One 重量级对象 Configuration 内部。在 Xml 映射文件中,
标签会被解析为 ParameterMap 对象,其每个子元素会被解析为 ParameterMapping 对
象。标签会被解析为 ResultMap 对象,其每个子元素会被解析为 ResultMapping 对象。每一个