- '#{}' 和'${}'的区别是什么?
${}是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常见的标签,除了<select><insert><update><delete>外,还有很多其他的标签,如<resultMap><parameterMap>、
<sql>、<include>、<selectKey>加上动态sql的9个标签trim|where|set|foreach|if|choose|when|otherwise|bind等,其中通过
<include>标签引入sql片段,<selectKey>为不支持自增的主键生成策略标签。
3.通常一个XML映射文件,都会写一个Dao接口与之对应,这个Dao接口的工作原理是什么?Dao接口里的方法,参数不同时,方法能重载吗?
Dao接口,也就是Mapper接口,接口的全限定名,就是映射文件中的namespace的值,接口的方法名,就是映射文件中
MappedStatement的id值,接口方法内的参数,就是传递给sql的参数。Mapper接口是没有实现类的,当调用接口方法的时候,接
口全限名+方法名拼接字符串作为key值,可唯一定位一个MappedStatement。在MyBatis中,每一个<select>、<insert>、
<update>、<delete>标签,都会被解析成一个MappedStatement对象。
Dao 接口里的方法可以重载,但是Mybatis的XML里面的ID不允许重复。
/**
* Mapper接口里面方法重载
*/
public interface StuMapper {
List<Student> getAllStu();
List<Student> getAllStu(@Param("id") Integer id);
}
//然后在 StuMapper.xml 中利用Mybatis的动态sql就可以实现。
<select id="getAllStu" resultType="com.pojo.Student">
select * from student
<where>
<if test="id != null">
id = #{id}
</if>
</where>
</select>
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执行结果封装为目标对象并返回的?都有哪些映射形式?
第一种是使用<resultMap>标签,逐一定义列名和对象属性名之间的映射关系。第二种是使用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<String,Statement>中,供下一次使用。
- 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>标签会被解析为 ParameterMap 对象,其每个子元素会被解析为 ParameterMapping 对
象。<resultMap>标签会被解析为 ResultMap 对象,其每个子元素会被解析为 ResultMapping 对象。每一个
<select>、<insert>、<update>、<delete>标签均会被解析为 MappedStatement 对象,标签内的 sql 会被解析为 BoundSql 对象。
17.为什么说 MyBatis 是半自动 ORM 映射工具?它与全自动的区别在哪里?
Hibernate 属于全自动 ORM 映射工具,使用 Hibernate 查询关联对象或者关联集合对象时,可以根据对象关
系模型直接获取,所以它是全自动的。而 MyBatis 在查询关联对象或关联集合对象时,需要手动编写 sql 来
完成,所以,称之为半自动 ORM 映射工具。
18.MyBatis的两种映射方式:
1.通过XML映射
2.通过注解
19.#{} 和 ${}的区别
#{}是预编译处理,${}是字符串替换。
Mybatis在处理#{}时,会将sql中的#{}替换为?号,调用PreparedStatement的set方法来赋值;
Mybatis在处理时就是把${}替换成变量的值。
使用#{}可以有效的防止SQL注入,提高系统安全性。
20.自增主键回填以及实现方法
新增一条数据成功后,将这条数据的主键封装到实体类中,并查看主键的值。
实现一:使用insert标签的子标签selectKey实现;
实现二:使用insert标签的属性useGeneratedKeys,keyProperty,keyColumn实现;
21.Mybatis的一级、二级缓存
一级缓存:(本地缓存)作用域:SqlSession级别
mybatis第一次查询到数据后,会将数据存入到一级缓存中。当再次查询相同的数据时,mybatis会将缓存中的数据直接拿出来返回,不再从数据库中查询;
二级缓存:(全局缓存) 作用域:namespace级别
在一个sqlSession,进行了一次查询,会把查询结果存入一级缓存;
当sqlSession关闭之后,mybatis会把一级缓存中的数据存入二级缓存中;
当我们再次发生相同的查询时,会从二级缓存中命中;
二级缓存作用域更大
二级缓存的开启需要进行配置,实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的。
22.MyBatis实现一对多查询有几种方式,怎么操作的?
关联对象查询,有两种实现方式,一种是单独发送一个 sql 去查询关联对象,赋给主对象,然后返回主对象。
另一种是使用嵌套查询,嵌套查询的含义为使用 join 查询,一部分列是 A 对象的属性值,另外一部分列是关
联对象 B 的属性值,好处是只发一个 sql 查询,就可以把主对象和其关联对象查出来。