MyBatis缓存

    技术2022-07-10  88

    MyBatis缓存

    参考资料:Mybatis 3官方文档.

    ​ 微信公众号:狂神说

    一、简介

    1、什么是缓存 [ Cache ]?

    存在内存中的临时数据。将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题。

    2、为什么使用缓存?

    减少和数据库的交互次数,减少系统开销,提高系统效率。

    3、什么样的数据能使用缓存?

    经常查询并且不经常改变的数据。

    二、Mybatis缓存

    MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率。

    MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存

    默认情况下,只有一级缓存开启。(SqlSession级别的缓存,也称为本地缓存)二级缓存需要手动开启和配置,他是基于namespace级别的缓存。为了提高扩展性,MyBatis定义了缓存接口Cache。我们可以通过实现Cache接口来自定义二级缓存

    2.1 一级缓存

    一级缓存(sqlSession级别的缓存)也叫本地缓存:

    与数据库同一次会话期间查询到的数据会放在本地缓存中。以后如果需要获取相同的数据,直接从缓存中拿,没必须再去查询数据库;

    测试:

    示例环境代码用Mybatis入门(一).

    开启日志

    <settings> <!--标准日志工厂实现--> <setting name="logImpl" value="STDOUT_LOGGING"/> <!--开启驼峰名命转换--> <setting name="mapUnderscoreToCamelCase" value="true"/> </settings> @Test public void test(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); UserMapper userMapper = sqlSession.getMapper(UserMapper.class); User user1 = userMapper.getUserById(1); User user2 = userMapper.getUserById(1); System.out.println(user2); System.out.println(user1==user2); sqlSession.close(); }

    结果分析:

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wkQCTFbl-1593529993895)(C:\Users\wangqun\AppData\Roaming\Typora\typora-user-images\image-20200630225007367.png)]

    一级缓存失效的四种情况

    sqlSession不同

    sqlSession相同,查询条件不同

    sqlSession相同,两次查询之间执行了增删改操作!

    sqlSession相同,手动清除一级缓存

    sqlSession.clearCache();//手动清除一级缓存

    2.2 二级缓存

    二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存

    基于namespace级别的缓存,一个名称空间,对应一个二级缓存;

    工作机制

    一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中;

    如果当前会话关闭了,这个会话对应的一级缓存就没了;但是我们想要的是,会话关闭了,一级缓存中的数据被保存到二级缓存中;

    新的会话查询信息,就可以从二级缓存中获取内容;

    不同的mapper查出的数据会放在自己对应的缓存(map)中;

    MyBatis 内置了一个强大的事务性查询缓存机制,它可以非常方便地配置和定制。 为了使它更加强大而且易于配置,我们对 MyBatis 3 中的缓存实现进行了许多改进。

    步骤:

    开启全局缓存(这个值默认是true,为了阅读方便显式开启) <!--显式开启全局缓存--> <setting name="cacheEnabled" value="true"/> 在要使用二级缓存的Mapper开启

    默认情况下,只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存。 要启用全局的二级缓存,只需要在你的 SQL 映射文件中添加一行:

    <cache/> <select id="getUserById" resultMap="UserMap" useCache="true"> select * from user where id = #{id} </select>

    cache属性修改

    <cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/> LRU – 最近最少使用:移除最长时间不被使用的对象。FIFO – 先进先出:按对象进入缓存的顺序来移除它们。SOFT – 软引用:基于垃圾回收器状态和软引用规则移除对象。WEAK – 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。

    结论

    a. 只要开启了二级缓存,我们在同一个Mapper中的查询,可以在二级缓存中拿到数据

    b. 查出的数据都会被默认先放在一级缓存中

    c. 只有会话提交或者关闭以后,一级缓存中的数据才会转到二级缓存中

    2.3 自定义缓存

    除了上述缓存的方式,你也可以通过实现你自己的缓存,或为其他第三方缓存方案创建适配器,来完全覆盖缓存行为。

    <cache type="com.domain.something.MyCustomCache"/>

    缓存原理图

    Processed: 0.041, SQL: 9