一文带你打通mybatis缓存

举报
yd_249383650 发表于 2023/07/30 17:02:02 2023/07/30
【摘要】 ​ 目录基本概述缓存:cache分类一级缓存demo演示讲解 什么情况下不走缓存?二级缓存 demo演示 二级缓存的相关配置基本概述缓存:cache缓存的作用:通过减少IO的方式,来提高程序的执行效率。MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制。缓存可以极大的提升查询效率。分类MyBatis系统中默认定义了两级缓存,分别是一级缓存和二级缓存。基本介绍:① 默认情...

 

目录


基本概述

缓存:cache

分类

一级缓存

demo演示讲解

 什么情况下不走缓存?

二级缓存

 demo演示

 二级缓存的相关配置



基本概述

缓存:cache

缓存的作用:通过减少IO的方式,来提高程序的执行效率。MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制。缓存可以极大的提升查询效率。

分类

MyBatis系统中默认定义了两级缓存,分别是一级缓存和二级缓存。

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

缓存只针对于DQL语句,也就是说缓存机制只对应select语句。

一级缓存

一级缓存默认是开启的。不需要做任何配置。

原理:只要使用同一个SqlSession对象执行同一条SQL语句,就会走缓存。

demo演示讲解

    public static void main(String[] args) {
        SqlSession session = SqlSessionUtil.openSession();
        StudentMapper mapper = session.getMapper(StudentMapper.class);
        //第一次查询
        List<Student> students = mapper.selectByCid(2101L);
        //第二次查询
        List<Student> students1 = mapper.selectByCid(2101L);
    }

上述代码使用同一个SqlSession去查询数据库,按照道理应该是查询俩次数据库,但由于缓存的存在,故只会查询一遍数据库,可见下面

编辑

 什么情况下不走缓存?

没有使用到当前一级缓存的情况,效果就是还要再向数据库发出查询

  • 第一种:不同的SqlSession对象。
  • 第二种:查询条件变化了。

一级缓存失效情况包括两种:

  • 第一种:第一次查询和第二次查询之间,手动清空了一级缓存。
sqlSession.clearCache();

注意,该方法只清除当前session的一级缓存。 

  • 第二种:第一次查询和第二次查询之间,执行了增删改操作。【这个增删改和哪张表没有关系,只要有insert delete update操作,一级缓存就失效。】

编辑

实际上,这个是因为每个增删改查都有标签flushCache,增删改默认为flushCache=“true”,即执行完后就清除一级缓存和二级缓存。  


二级缓存

二级缓存的范围是SqlSessionFactory。

使用二级缓存需要具备以下几个条件:

  1. <setting name="cacheEnabled" value="true"> 全局性地开启或关闭所有映射器配置文件中已配置的任何缓存。默认就是true,无需设置。
  2. 在需要使用二级缓存的SqlMapper.xml文件中添加配置:<cache />
  3. 使用二级缓存的实体类对象必须是可序列化的,也就是必须实现java.io.Serializable接口
  4. SqlSession对象关闭或提交之后,一级缓存中的数据才会被写入到二级缓存当中。此时二级缓存才可用。

数据会从二级缓存中取出。查出的数据都会被默认先放在一级缓存中,只有会话提交或者关闭以后,一级缓存中的数据才会转移到二级缓存中。

 demo演示

@Test
public void testSelectById2() throws Exception{
    SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));

    SqlSession sqlSession1 = sqlSessionFactory.openSession();
    CarMapper mapper1 = sqlSession1.getMapper(CarMapper.class);
    Car car1 = mapper1.selectById(83L);
    System.out.println(car1);

    // 关键一步
    sqlSession1.close();

    SqlSession sqlSession2 = sqlSessionFactory.openSession();
    CarMapper mapper2 = sqlSession2.getMapper(CarMapper.class);
    Car car2 = mapper2.selectById(83L);
    System.out.println(car2);
}

 编辑


 二级缓存的失效:只要两次查询之间出现了增删改操作。二级缓存就会失效。【一级缓存也会失效】

 二级缓存的相关配置

编辑

 evictio:缓存的回收策略

    LRU - 最近最少使用的,移除最长时间不被使用的对象。
    FIFO - 先进先出,按对象进入缓存的顺序来移除他们。
    SOFT - 软引用,移除基于垃圾回收器状态和引用规则的对象。
    WEAK - 弱引用,更积极的移除基于垃圾收集器状态和弱引用规则的对象。默认的是LRU。

flushInterval:缓存刷新间隔

    缓存多长时间清空一次,默认不清空,可设置一个毫秒值。

readOnly:是否只读

    true:只读,MyBatis认为所有从缓存中获取数据的操作都是只读操作,不会修改数据,所以为了加快速度,mybatis直接将数据在缓存中的引用交给用户。这种方式不安全,但速度快。
    false:非只读,MyBatis觉得获取的数据肯能会被修改,所以MyBatis会利用序列化和反序列化的技术克隆一份新的数据给你。这种方式安全,但是速度慢。因此,此时需要将所有需要在二级缓存中存放的对象都实现序列化接口。

size:缓存存放多少元素
type:指定自定义缓存的全类名


【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。