框架源码专题:Mybatis的一级缓存、二级缓存是什么?有什么作用?

1. Mybatis中缓存的作用

首先缓存和数据库DB都是用来存储数据的,那么他们有什么区别呢?

        数据库是关系型数据库的,用来存储复杂类型的数据到磁盘上,可以完成数据持久化!而缓存的数据是存在内存中的,当断电或者其他情况时,缓存中的数据是会丢失的,无法做持久化。随着缓存技术的发展,部分缓存也实现了持久化,比如redis,即便如此,数据库持久化的地位仍无可撼动!也正因为缓存中的数据是存在内存中的,所以获取数据的速度比从数据库中获取数据速度要快。

        现在应用基本上是缓存和数据库结合起来使用,目的是为了减少数据库的访问压力。当大量用户请求过来时,先经过缓存,缓存没有再查数据库,然后同步缓存,下次用户再访问时,经过缓存就可以直接获取数据。减轻了数据库的压力,当然也不是所有的数据都会放入缓存中,一般会放入以下三种类型的数据:

  • ①:变化较小的数据,比如字典型数据
  • ②:并发量比较大的数据
  • ③:分布式锁、分布式session
  • ④:临时数据等等…

        Mybatis缓存分为一级缓存和二级缓存,目的就是减少数据库压力,其中二级缓存使用到了装饰器和责任链的设计模式,这也是优秀框架基本都带有缓存实现的原因!!

        在MyBatis中,Cache是缓存接口,定义了一些基本的缓存操作,所有缓存类都应该实现该接口。MyBatis内部提供了丰富的缓存实现类,比如:

  • 具有基本缓存功能的PerpetualCache
  • 具有LRU策略的缓存LruCache
  • 可保证线程安全的缓存SynchronizedCache
  • 具备阻塞功能的缓存BlockingCache


2. 一级缓存

2.1 Mybatis的一级缓存特性:

  1. 默认开启一级缓存,也可以通过设置settings属性localCacheScope = STATEMENT 进行关闭
设置名描述有效值默认值
localCacheScopeMyBatis 利用本地缓存机制(Local Cache)防止循环引用和加速重复的嵌套查询。 默认值为 SESSION,会缓存一个会话中执行的所有查询。 若设置值为 STATEMENT,本地缓存将仅用于执行语句,对相同 SqlSession 的不同查询将不会进行缓存。SESSION 或STATEMENTSESSION
  1. 作用域是一个sqlSession
  2. 一级缓存默认实现类PerpetualCache,使用map存储缓存数据

在这里插入图片描述
map的key = hashcode + sqlid + hashcode + environments默认id,具备唯一性
map的value = 数据结果
在这里插入图片描述


2.2 一级缓存的失效情况

  1. 不同的sqlSession会使一级缓存失效
  2. 同一个sqlSession,但查询语句不一样,因为缓存的key不同了
  3. 同一个sqlSession,查询一句一样,但中间执行过增删改操作
  4. 同一个sqlSession,查询一句一样,但中间执行过手动清除缓存操作 session.clearCache()


3. 二级缓存

3.1 二级缓存特性

  1. 默认开启,但是没有实现。只有在需要使用到二级缓存的Mapper映射文件中加入cache标签,二级缓存才起作用

    <cache></cache>

  2. 作用域:基于全局范围,应用级别。

  3. 二级缓存默认实现类也是PerpetualCache,使用map进行存储。如果配置的缓存是默认类型,则mybatis会根据配置自动追加一系列装饰器。装饰顺序如下:SynchronizedCache–>LoggingCache–>SerializedCache–>ScheduledCache–>LruCache–>PerpetualCache

  4. 存数据时机:sqlSession提交的时候才往二级缓存中存数据

  5. 取数据时机:取数据时先从二级缓存中取,如果没有再去一级缓存中取。

  6. 使用到二级缓存的javaBean需要实现序列化接口 implements Serializable


3.2 二级缓存失效时机

  1. 同一个命名空间进行了增删改的操作,会导致二级缓存失效。

3.3 二级缓存的局限性

Mybatis的二级缓存在使用中还是存在一定缺陷的!比如当UserMapper.xml中的sql语句进行了多表查询,关联了一个部门Dept表,此时可以使用 " cache-ref " 标签来解决

 <cache-ref namespace="cn.tulingxueyuan.mapper.DeptMapper"/>

但是如果UserMapper.xml中关联了多张表,比如Oder订单表 、Dept部门表 、Member会员表等等,这种情况Mybatis的缓存就无能为力了。然后大部分场景都是多表关联查询,所以一般不会使用Mybatis的缓存,而是用redis等替代方案!

3.4 cache标签 的属性解析

   		 <cache
            type="org.mybatis.caches.redis.RedisCache"
            eviction="FIFO"
            flushInterval="60000"
            size="512"
            readOnly="false"/>

①:type 二级缓存实现类 org.apache.ibatis.cache.impl.PerpetualCache(默认可省略)。

②: eviction 当缓存达到一定的数量的时候(size的数量),以哪种方式进行回收缓存:

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

③:flushInterval 刷新时间毫秒
可以被设置为任意的正整数,设置的值应该是一个以毫秒为单位的合理时间量。 默认情况是不设置,也就是没有刷新间隔,缓存仅仅会在调用语句时刷新。

④: size 可以缓存的数量
可以被设置为任意正整数,要注意欲缓存对象的大小和运行环境中可用的内存资源。默认值是 1024。

⑤: readOnly 设置数据是否只读

          true       读取缓存的数据性能更高,会直接获取内存引用
          false 默认 读取缓存的数据性能更低,会将数据拷贝一份


4. 一级缓存和二级缓存的区别

4.1 一级缓存和二级缓存分别是什么时候进行存储数据的?

一级缓存:在查询完就会存储!
二级缓存:在事务提交的时候才会存储!

4.2 从缓存中取数据时的顺序是怎样的?

取数据时会先去二级缓存中取,如果二级缓存中没有,再去一级缓存中拿!

4.3 一二级缓存的数据类型是怎样的?

一级缓存:使用Map存储,存储的是xxxMapper.xml文件中的每一个增删改查节点,key 根据sqlID来区分,value = 查询结果

二级缓存:相对于一级缓存做了一次包装,key = xxxMapper.xml的命名空间,value = 类似于一级缓存Map结构! 因为只有标识了“cache”标签的xml文件才会使用到二级缓存,不同的Mapper.xml要根据二级缓存的key来区别(也就是 xxxMapper.xml的命名空间)!


5. 通过代码观察Mybatis缓存工作的全过程

通过以下两次查询代码 观察缓存

   // 通过加载配置文件流构建一个SqlSessionFactory  DefaultSqlSessionFactory,将XML配置文件构建为Configuration配置类
   SqlSessionFactory sqlMapper = new SqlSessionFactoryBuilder().build(reader);
   // 获取SqlSession 
   SqlSession session = sqlMapper.openSession();
 
   //第一次查询
   UserMapper mapper = session.getMapper(UserMapper.class);
   User user2 = mapper.selectById(1L);
   System.out.println("user2 = "+user2);

   //session提交
   session.commit();

	//第二次查询
   UserMapper mapper1 = session.getMapper(UserMapper.class);
   User user21 = mapper1.selectById(1L);
   System.out.println("user21 = "+user21);

在获取到SqlSession后,代码运行流程如下图

在这里插入图片描述
缓存命中率,是由 缓存命中次数 / 查询次数 所得到的值
打印结果如下:
在这里插入图片描述

如果去掉代码中的session.commit(); 那么在两次查询中,只会从一级缓存中获取到数据,因为SqlSession没有提交,二级缓存就不会有数据,两次打印缓存命中率结果应都是: Cache Hit Ratio = 0

验证:
去掉session.commit() 后,终端打印如下:
在这里插入图片描述

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值