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常见的标签,除了<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实现;


33.PNG

21.Mybatis的一级、二级缓存

一级缓存:(本地缓存)作用域:SqlSession级别

mybatis第一次查询到数据后,会将数据存入到一级缓存中。当再次查询相同的数据时,mybatis会将缓存中的数据直接拿出来返回,不再从数据库中查询;

二级缓存:(全局缓存) 作用域:namespace级别

在一个sqlSession,进行了一次查询,会把查询结果存入一级缓存;
当sqlSession关闭之后,mybatis会把一级缓存中的数据存入二级缓存中;
当我们再次发生相同的查询时,会从二级缓存中命中;
二级缓存作用域更大
二级缓存的开启需要进行配置,实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的。

22.MyBatis实现一对多查询有几种方式,怎么操作的?

关联对象查询,有两种实现方式,一种是单独发送一个 sql 去查询关联对象,赋给主对象,然后返回主对象。
另一种是使用嵌套查询,嵌套查询的含义为使用 join 查询,一部分列是 A 对象的属性值,另外一部分列是关
联对象 B 的属性值,好处是只发一个 sql 查询,就可以把主对象和其关联对象查出来。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容