MyBatis

目录

什么是 MyBatis?

 讲下 MyBatis 的缓存

Mybatis 是如何进行分页的?分页插件的原理是什么?

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

Mybatis 动态 sql 是做什么的?都有哪些动态 sql?能简述一下动态 sql 的执行原理不?

#{}和${}的区别是什么?

为什么说 Mybatis 是半自动 ORM 映射工具?它与全自动的区别在哪里?

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

MyBatis 与 Hibernate 有哪些不同?

MyBatis 的好处是什么?

什么是 MyBatis 的接口绑定,有什么好处?

接口绑定有几种实现方式,分别是怎么实现的?

什么情况下用注解绑定,什么情况下用 xml 绑定?

MyBatis 实现一对一有几种方式?具体怎么操作的?

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

MyBatis 里面的动态 Sql 是怎么设定的?用什么语法?

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

Xml 映射文件中,除了常见的 select|insert|updae|delete 标签之外,还有哪些标签?

当实体类中的属性名和表中的字段名不一样,如何将查询的结果封装到指定 pojo?

模糊查询 like 语句该怎么写

通常一个 Xml 映射文件,都会写一个 Dao 接口与之对应, Dao 的工作原理,是否可以重载?

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

Mybatis 的 Xml 映射文件中,不同的 Xml 映射文件,id 是否可以重复?

Mybatis 中如何执行批处理?

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

Mybatis 是否可以映射 Enum 枚举类?

在 mapper 中如何传递多个参数?

resultType resultMap 的区别?

使用 MyBatis 的 mapper 接口调用时有哪些要求?

Mybatis 比 IBatis 比较大的几个改进是什么?

IBatis 和 MyBatis 在核心处理类分别叫什么?

IBatis 和 MyBatis 在细节上的不同有哪些?


什么是 MyBatis?

MyBatis 是一个可以自定义 SQL 、存储过程和高级映射的持久层框架。

 讲下 MyBatis 的缓存

MyBatis 的缓存分为一级缓存和二级缓存 , 一级缓存放在 session 里面 , 默认就有 , 二级缓
存放在它的命名空间里 , 默认是不打开的 , 使用二级缓存属性类需要实现 Serializable 序列化
接口 ( 可用来保存对象的状态 ), 可在它的映射文件中配置 <cache/>

Mybatis 是如何进行分页的?分页插件的原理是什么?

1 Mybatis 使用 RowBounds 对象进行分页,也可以直接编写 sql 实现分页,也可以使用
Mybatis 的分页插件。
2 )分页插件的原理:实现 Mybatis 提供的接口,实现自定义插件,在插件的拦截方法内拦
截待执行的 sql ,然后重写 sql
举例: select * from student ,拦截 sql 后重写为: select t.* from select * from student t
limit 0 10

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

1 Mybatis 仅可以编写针对 ParameterHandler ResultSetHandler StatementHandler
Executor 4 种接口的插件, Mybatis 通过动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这 4 种接口对象的方法时,就会进入拦截方法,具体就是 InvocationHandler 的 invoke() 方法,当然,只会拦截那些你指定需要拦截的方法。
2 )实现 Mybatis Interceptor 接口并复写 intercept() 方法,然后在给插件编写注解,指定
要拦截哪一个接口的哪些方法即可,记住,别忘了在配置文件中配置你编写的插件。

Mybatis 动态 sql 是做什么的?都有哪些动态 sql?能简述一下动态 sql 的执行原理不?

1 Mybatis 动态 sql 可以让我们在 Xml 映射文件内,以标签的形式编写动态 sql ,完成逻辑判断和动态拼接 sql 的功能。
2 Mybatis 提供了 9
动态 sql 标签:    trim|where|set|foreach|if|choose|when|otherwise|bind
3 )其执行原理为,使用 OGNL sql 参数对象中计算表达式的值,根据表达式的值动态拼
sql ,以此来完成动态 sql 的功能。

#{}和${}的区别是什么?

1 #{} 是预编译处理, ${} 是字符串替换。
2 Mybatis 在处理 #{} 时,会将 sql 中的 #{} 替换为 ? 号,调用 PreparedStatement set 方法来赋值;
3 Mybatis 在处理 ${} 时,就是把 ${} 替换成变量的值。
4 )使用 #{} 可以有效的防止 SQL 注入,提高系统安全性。

为什么说 Mybatis 是半自动 ORM 映射工具?它与全自动的区别在哪里?

Hibernate 属于全自动 ORM 映射工具,使用 Hibernate 查询关联对象或者关联集合对象
时,可以根据对象关系模型直接获取,所以它是全自动的。而 Mybatis 在查询关联对象或
关联集合对象时,需要手动编写 sql 来完成,所以,称之为半自动 ORM 映射工具。

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

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

MyBatis 与 Hibernate 有哪些不同?

1 Mybatis hibernate 不同,它不完全是一个 ORM 框架,因为 MyBatis 需要程序员自己 编写 Sql 语句,不过 mybatis 可以通过 XML 或注解方式灵活配置要运行的 sql 语句,并将 java 对象和 sql 语句映射生成最终执行的 sql ,最后将 sql 执行的结果再映射生成 java 对 象。
2 Mybatis 学习门槛低,简单易学,程序员直接编写原生态 sql ,可严格控制 sql 执行性
能,灵活度高,非常适合对关系数据模型要求不高的软件开发,例如互联网软件、企业运
营类软件等,因为这类软件需求变化频繁,一但需求变化要求成果输出迅速。但是灵活的
前提是 mybatis 无法做到数据库无关性,如果需要实现支持多种数据库的软件则需要自定
义多套 sql 映射文件,工作量大。
3 Hibernate 对象 / 关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如
需求固定的定制化软件)如果用 hibernate 开发可以节省很多代码,提高效率。但是
Hibernate 的缺点是学习门槛高,要精通门槛更高,而且怎么设计 O/R 映射,在性能和对象模型之间如何权衡,以及怎样用好 Hibernate 需要具有很强的经验和能力才行。
总之,按照用户的需求在有限的资源环境下只要能做出维护性、扩展性良好的软件架构都
是好架构,所以框架只有适合才是最好。

MyBatis 的好处是什么?

1 MyBatis sql 语句从 Java 源程序中独立出来,放在单独的 XML 文件中编写,给程序的 维护带来了很大便利。
2 MyBatis 封装了底层 JDBC API 的调用细节,并能自动将结果集转换成 Java Bean 对象, 大大简化了 Java 数据库编程的重复工作。
3 )因为 MyBatis 需要程序员自己去编写 sql 语句,程序员可以结合数据库自身的特点灵活
控制 sql 语句,因此能够实现比 Hibernate 等全自动 orm 框架更高的查询效率,能够完成复杂查询。

什么是 MyBatis 的接口绑定,有什么好处?

接口映射就是在 MyBatis 中任意定义接口 , 然后把接口里面的方法和 SQL 语句绑定 , 我们
直接调用接口方法就可以 , 这样比起原来了 SqlSession 提供的方法我们可以有更加灵活的选择和设置.

接口绑定有几种实现方式,分别是怎么实现的?

接口绑定有两种实现方式, 一种是通过注解绑定 , 就是在接口的方法上面加上
@Select@Update 等注解里面包含 Sql 语句来绑定 , 另外一种就是通过 xml 里面写 SQL 来绑 定, 在这种情况下 , 要指定 xml 映射文件里面的 namespace 必须为接口的全路径名 .

什么情况下用注解绑定,什么情况下用 xml 绑定?

Sql 语句比较简单时候 , 用注解绑定;当 SQL 语句比较复杂时候 , xml 绑定 , 一般用
xml 绑定的比较多

MyBatis 实现一对一有几种方式?具体怎么操作的?

有联合查询和嵌套查询 ,
联合查询是几个表联合查询 , 只查询一次 , 通过在 resultMap 里面
配置 association 节点配置一对一的类就可以完成 ;
嵌套查询是先查一个表 , 根据这个表里面的结果的外键 id, 去再另外一个表里面查询数据 , 也是通过 association 配置 , 但另外一个表的查询通过 select 属性配置。

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

能, Mybatis 不仅可以执行一对一、一对多的关联查询,还可以执行多对一,多对多的

关联查询,多对一查询,其实就是一对一查询

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

MyBatis 里面的动态 Sql 是怎么设定的?用什么语法?

MyBatis 里面的动态 Sql 一般是通过 if 节点来实现 , 通过 OGNL 语法来实现 , 但是如果要
写的完整 , 必须配合 where,trim 节点 ,where 节点是判断包含节点有内容就插入 where, 否则不 插入,trim 节点是用来判断如果动态语句是以 and or 开始 , 那么会自动把这个 and 或者 or取掉。

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

第一种是使用 <resultMap> 标签,逐一定义列名和对象属性名之间的映射关系。
第二种是使用 sql 列的别名功能,将列别名书写为对象属性名,比如 T_NAME AS NAME ,对
象属性名一般是 name ,小写,但是列名不区分大小写, Mybatis 会忽略列名大小写,智能
找到与之对应对象属性名,你甚至可以写成 T_NAME AS NaMe Mybatis 一样可以正常工
作。
有了列名与属性名的映射关系后, Mybatis 通过反射创建对象,同时使用反射给对象的属性
逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。

Xml 映射文件中,除了常见的 select|insert|updae|delete 标签之外,还有哪些标签?

还有很多其他的标签, <resultMap> <parameterMap> <sql> <include>
<selectKey> ,加上动态 sql 9 个标签,
trim|where|set|foreach|if|choose|when|otherwise|bind 等,其中 <sql> sql 片段标签,通
<include> 标签引入 sql 片段, <selectKey> 为不支持自增的主键生成策略标签。

当实体类中的属性名和表中的字段名不一样,如何将查询的结果封装到指定 pojo?

1 )通过在查询的 sql 语句中定义字段名的别名。
2 )通过 <resultMap> 来映射字段名和实体类属性名的一一对应的关系。

模糊查询 like 语句该怎么写

select * from table1 where name like '*明*'

  1. % 表示零个或多个字符的任意字符串
  2. _(下划线)表示任何单个字符
  3. [ ] 表示指定范围 ([a-f]) 或集合 ([abcdef]) 中的任何单个字符:

通常一个 Xml 映射文件,都会写一个 Dao 接口与之对应, Dao 的工作原理,是否可以重载?

不能重载,因为通过 Dao 寻找 Xml 对应的 sql 的时候全限名 + 方法名的保存和寻找策略。
接口工作原理为 jdk 动态代理原理,运行时会为 dao 生成 proxy ,代理对象会拦截接口方法,去执行对应的 sql 返回数据。

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

虽然 Mybatis 解析 Xml 映射文件是按照顺序解析的,但是,被引用的 B 标签依然可以
定义在任何地方, Mybatis 都可以正确识别。
原理是, Mybatis 解析 A 标签,发现 A 标签引 用了 B 标签,但是 B 标签尚未解析到,尚不存在,此时, Mybatis 会将 A 标签标记为未解析状态,然后继续解析余下的标签,包含 B 标签,待所有标签解析完毕, Mybatis 会重新 解析那些被标记为未解析的标签,此时再解析 A 标签时, B 标签已经存在, A 标签也就可以正常解析完成了。

Mybatis 的 Xml 映射文件中,不同的 Xml 映射文件,id 是否可以重复?

不同的 Xml 映射文件,如果配置了 namespace ,那么 id 可以重复;如果没有配置
namespace ,那么 id 不能重复;毕竟 namespace 不是必须的,只是最佳实践而已。原因就
namespace+id 是作为 Map<String>, MappedStatement>的 key 使用的,如果namespace,就剩下 id ,那么, id 重复会导致数据互相覆盖。有了 namespace ,自然 id 就可以重复,namespace 不同, namespace+id 自然也就不同。

Mybatis 中如何执行批处理?

Foreach方式:
当数据过多时,可能生成的动态sql过大,mysql默认仅1M的sql字符串,过长可能会执行失败,可以通过修改配置文件,batch方式无限制。在sql循环时设置batch与否其实执行时间差别不是很大,几乎可以忽略不计。所以其实如果不是特别要求性能。可以直接在sql中使用for循环即可。
Batch方式:
如果需要使用batch,请在需要的函数上面设置batch,不要全局使用。因为batch也是有副作用的。比如在Insert操作时,在事务没有提交之前,是没有办法获取到自增的id,;此外,对于update、delete无法返回更新、插入条数。这在某型情形下是不符合业务要求

批处理选择
已知sql执行效率:insert > delete > update,因此对应的批处理也是一样的顺序。
场景示例:批处理,若数据表未存在记录则作插入,若已存在则作更新。
方案一:使用select + update + insert做批处理,但此方案效率较低。
方案二:使用select + delete + insert 做批处理,推荐使用。

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

Executor执行器,是mybatis中执行查询的主要代码,可以在mybatis的配置文件中设置使用哪种执行器,Executor分为三种,分别是

简单执行器SimpleExecutor、

可重用执行器ReuseExecutor、

批量执行器BatchExecutor。

1)SimpleExecutor :每执行一次 update select ,就开启一个 Statement
象,用完立刻关闭 Statement 对象。
2 ReuseExecutor :执行 update select ,以 sql 作为key 查找 Statement 对象,存在就使用,不存在就创建,用完后,不关闭 Statement 对象,而是放置于 Map3
3)BatchExecutor:完成批处理。

  

Mybatis 是否可以映射 Enum 枚举类?

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

在 mapper 中如何传递多个参数?

1 )直接在方法中传递参数, xml 文件用 #{0} #{1} 来获取
2 )使用 @param 注解 : 这样可以直接在 xml 文件中通过 #{name} 来获取

resultType resultMap 的区别?

如果查询出来的结果的列名和实体属性不一致,通过定义一个 resultMap对列名和pojo属性名之间作一个映射关系

使用resultType进行输出映射,只有查询结果显示的列名和实体的属性名一致时,该列才可以映射成功。

使用 MyBatis 的 mapper 接口调用时有哪些要求?

1 Mapper 接口方法名和 mapper.xml 中定义的每个 sql id 相同
2 Mapper 接口方法的输入参数类型和 mapper.xml 中定义的每个 sql parameterType
类型相同
3 Mapper 接口方法的输出参数类型和 mapper.xml 中定义的每个 sql resultType 的类型
相同
4 Mapper.xml 文件中的 namespace 即是 mapper 接口的类路径。

Mybatis 比 IBatis 比较大的几个改进是什么?

1 )有接口绑定 , 包括注解绑定 sql xml 绑定 Sql
2 )动态 sql 由原来的节点配置变成 OGNL 表达式
3 )在一对一, 一对多的时候association,在一对多的时候引入了 collection 节点,不过都是在 resultMap 里面配置

IBatis 和 MyBatis 在核心处理类分别叫什么?

IBatis 里面的核心处理类交 SqlMapClient,MyBatis 里面的核心处理类叫做 SqlSession

IBatis 和 MyBatis 在细节上的不同有哪些?

1 )在 sql 里面变量命名有原来的 # 变量 # 变成了 #{ 变量 }
2 )原来的 $ 变量 $ 变成了 ${ 变量 }
3 )原来在 sql 节点里面的 class 都换名字叫  type
4 )原来的 queryForObject queryForList 变成了 selectOne selectList
5)原来的别名设置在映射文件里面放在了核心配置文件里
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值