面试官让我说一下Mybatis一级缓存、二级缓存的区别,史诗级回答让他刮目相看

88 篇文章 10 订阅
4 篇文章 1 订阅

说起mybatis,大家可能都知道它是一个优秀的久层框架,它支持定制化SQL、存储过程以及高级映射。面试中都会问起mybatis一级缓存和二级缓存,它体现出你对mybatis这个开发中的理解,如果照着答案背的话只能拿到一个及格分,所以今天咱们就好好聊聊mybatis。
在这里插入图片描述

另外本人整理了20年面试题大全,包含spring、并发、数据库、Redis、分布式、dubbo、JVM、微服务等方面总结,下图是部分截图,需要的话点这里点这里,暗号CSDN。

在这里插入图片描述

1.首先,什么是Mybatis?

  • MyBatis 是一个可以自定义 SQL、存储过程和高级映射的持久层框架。MyBatis 避免了几乎所有的 JDBC代码和手动设置参数以及获取结果集。

  • 我们在使用了 MyBatis 之后,只需要提供 SQL 语句就好了,其余的诸如:建立连接、操作 Statment、ResultSet,处理 JDBC 相关异常等等都可以交给 MyBatis 去处理,我们的关注点于是可以就此集中在 SQL 语句上,关注在增删改查这些操作层面上。

2.mybatis的框架

在这里插入图片描述

3.重点来了,讲下 MyBatis 的缓存

在这里插入图片描述

Mybatis对缓存提供支持,一级缓存是默认使用的,二级缓存需要手动开启。

区别:

  • 一级缓存的作用域是一个sqlsession内;
  • 二级缓存作用域是针对mapper进行缓存;

一级缓存:

在参数和SQL完全一样的情况下,我们使用同一个SqlSession对象调用一个Mapper方法,往往只执行一次SQL,因为使用SelSession第一次查询后,MyBatis会将其放在缓存中,以后再查询的时候,如果没有声明需要刷新,并且缓存没有超时的情况下,SqlSession都会取出当前缓存的数据,而不会再次发送SQL到数据库。
一级缓存简单示意图
一级缓存时执行commit,close,增删改等操作,就会清空当前的一级缓存;当对SqlSession执行更新操作(update、delete、insert)后并执行commit时,不仅清空其自身的一级缓存(执行更新操作的效果),也清空二级缓存(执行commit()的效果)。

二级缓存:

二级缓存指的就是同一个namespace下的mapper,二级缓存中,也有一个map结构,这个区域就是一级缓存区域。一级缓存中的key是由sql语句、条件、statement等信息组成一个唯一值。一级缓存中的value,就是查询出的结果对象。

在这里插入图片描述
1、在配置文件中 开启二级缓存的总开关

<setting name="cacheEnabled" value="true" />

2、 在mapper映射文件中开启二级缓存

<cache eviction="FIFO" flushInterval="60000" size="512" 
readOnly="true"/>

参数名属性eviction收回策略flushInterval刷新间隔size引用数目readOnly只读

关于eviction的各个参数属性:

  • 参数名属性eviction="LRU"最近最少使用的:移除最长时间不被使用的对象。
  • (默认)eviction="FIFO"先进先出:按对象进入缓存的顺序来移除它们。
  • eviction="SOFT"软引用:移除基于垃圾回收器状态和软引用规则的对象。
  • eviction="WEAK"弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。

3、实体类实现Serializable

禁用缓存

如测试sql语句性能时缓存会影响测试准确性 需要禁用在映射文件中:默认值是true useCache=”false”

<select id="findAllPets" resultMap="petsMap" useCache="false">
 select * from pets
</select>

刷新缓存

在映射文件中:属性:flushCache=”true”刷新缓存,在查询语句中,默认值是false,在新增删除修改语句中,默认值是true(清空缓存)

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

答:

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

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

答:

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

6.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()方法的调用。这就是延迟加载的基本原理。

7.MyBatis 与 Hibernate 有哪些不同?

答:

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

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

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

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

答:
第一种是使用标签,逐一定义列名和对象属性名之间的映射关系。
第二种是使用 sql 列的别名功能,将列别名书写为对象属性名,比如 T_NAME AS NAME,对象属性名一般是 name,小写,但是列名不区分大小写,Mybatis 会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成 T_NAME AS NaMe,Mybatis 一样可以正常工作。

有了列名与属性名的映射关系后,Mybatis 通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。

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

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

11.模糊查询 like 语句该怎么写

答:
1)在 java 中拼接通配符,通过#{}赋值
2)在 Sql 语句中拼接通配符 (不安全 会引起 Sql 注入)

最后:

针对最近很多人都在面试,我这边也整理了相当多的面试专题资料,也有其他大厂的面经。希望可以帮助到大家。

下面的面试题答案都整理成文档笔记。也还整理了一些面试资料&最新2020收集的一些大厂的面试真题(都整理成文档,小部分截图),有需要的可以点击进入暗号CSDN

在这里插入图片描述
最后的惯例:
在这里插入图片描述

  • 127
    点赞
  • 791
    收藏
    觉得还不错? 一键收藏
  • 18
    评论
### 回答1: 关于MyBatis一级缓存二级缓存,它们都是用来优化数据库访问性能的。一级缓存是在同一个 SqlSession 中进行缓存,可以避免重复查询同一个对象的数据,提高查询性能。二级缓存是在不同 SqlSession 之间共享的缓存,可以避免重复查询同一个表的数据,提高查询性能。但需要注意的是,二级缓存要保证缓存数据的一致性,需要使用配置文件中的缓存刷新策略进行设置。 ### 回答2: Mybatis是一种开源的持久层框架,用于将数据库操作映射到Java对象。在面试中如何回答Mybatis一级缓存二级缓存问题呢? 一级缓存是指Mybatis在执行数据库查询的过程中,将查询结果缓存在SqlSession对象中。当多次执行相同的查询时,Mybatis会首先检查一级缓存中是否存在该查询的结果,如果存在则直接返回缓存的结果,而不再去查询数据库。一级缓存的作用范围仅限于同一个SqlSession对象,当SqlSession对象关闭或执行更新、插入、删除等操作时,一级缓存会失效。 二级缓存是指Mybatis在执行数据库查询的过程中,将查询结果缓存在SessionFactory对象中。这意味着多个SqlSession对象可以共享同一个二级缓存。当多个SqlSession对象执行相同的查询时,首先会检查二级缓存中是否存在该查询的结果,如果存在则直接返回缓存的结果。二级缓存的作用范围跨越多个SqlSession对象,当整个应用程序关闭时,二级缓存会失效。 在回答这个问题时,可以从以下几个方面展开: 1. 对于一级缓存: - 首先简要介绍一级缓存的概念和特点; - 强调一级缓存的有效范围仅限于同一个SqlSession对象; - 提到当SqlSession对象关闭或执行更新操作时,一级缓存会失效。 2. 对于二级缓存: - 首先简要介绍二级缓存的概念和特点; - 强调多个SqlSession对象可以共享同一个二级缓存; - 提到当整个应用程序关闭时,二级缓存会失效。 3. 对比一级缓存二级缓存: - 强调一级缓存的作用范围较小,而二级缓存的作用范围跨越多个SqlSession对象; - 提到一级缓存是默认开启的,而二级缓存需要手动配置; - 适当提到一级缓存适用于高频度读取操作,而二级缓存适用于缓存经常被读取的数据。 最后,可以以自己的理解对Mybatis缓存的优势和适用场景进行总结,展示对Mybatis缓存的深入理解和实际运用经验。 ### 回答3: MyBatis是一种流行的Java持久化框架,提供了一级缓存二级缓存来提升性能。在面试中,回答如下: 一级缓存MyBatis一级缓存是指在同一个SqlSession内部的缓存。当在同一个SqlSession中执行相同的查询时,MyBatis会首先从缓存中查询结果,而不会去数据库查询。这大大提高了查询的性能。默认情况下,MyBatis开启了一级缓存二级缓存MyBatis二级缓存是指在不同的SqlSession之间共享的缓存。当在一个SqlSession中执行查询后,查询结果会被缓存下来,当再次执行相同的查询时,MyBatis会从二级缓存中获取结果,而不会再次去数据库查询。因为二级缓存是跨SqlSession的,所以可以在不同的SqlSession中共享查询结果,进一步提高了性能。需要注意的是,二级缓存是需要手动开启和配置的。 在面试中,可以回答如下的点来介绍一级缓存二级缓存: - 一级缓存是SqlSession别的缓存,可以提高同一个SqlSession中相同查询的性能。 - 二级缓存是共享的缓存,可以提高跨SqlSession的查询性能。 - 一级缓存默认开启且无需手动配置,而二级缓存需要手动配置开启。 - 一级缓存是默认的缓存别,而二级缓存是可选的。 - 一级缓存的生命周期是SqlSession别的,而二级缓存的生命周期是应用别的。 - 一级缓存二级缓存可以提高查询性能,但在并发环境下需要注意缓存的同步和数据一致性,避免出现脏读问题。 在面试中,除了回答上述内容,还可以结合自己的实际项目经验,分享在使用MyBatis缓存时遇到的问题和解决方案,以及在配置和使用缓存时的注意事项等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值