Mybatis第三天 缓存

本文详细解释了MyBatis中的一级缓存(会话级别)和二级缓存(mapper级别)的概念、适用场景、命中条件以及相关属性。一级缓存在SqlSession中存储查询结果,二级缓存则针对表的查询结果共享给所有用户。
摘要由CSDN通过智能技术生成

1.缓存的概念

    1.1什么是缓存?

        缓存就是内存中的数据,常常来自对数据库查询结果的保存。使用缓存,我们可以避免频繁与数据库进行交互,从而提高响应速度。

    1.2为什么要使用缓存?

        减少和数据库的交互次数,提高执行效率。(因为查询数据库是一件很费时很费效率的事,还涉及一些硬盘等io操作,而缓存是存在内存中的,读取都很快,而且效率高)

    1.3什么样的数据能使用缓存,什么样的数据不能使用

          适用于缓存:

                1.经常查询并且不经常改变的。

                2.数据的正确与否对最终结果影响不大的。

          不适用于缓存:

                1.经常改变的数据

                2.数据的正确与否对最终结果影响很大的。例如:商品的库存,银行的汇率,股市的牌价。

一级缓存

    什么是一级缓存

        MyBatis 包含了一个非常强大的查询缓存特性,它可以非常方便地配置和定制。MyBatis 3 中的缓存实现的很多改进都已经实现了,使得它更加强大而且易于配置。mybatis 默认情况下只会开启一级缓存,也就是局部的 session 会话缓存

首先我们要知道什么是查询缓存?查询缓存又有什么作用?

如下图,每一个 session 会话都会有各自的缓存,这缓存是局部的,也就是所谓的一级缓存: 

1.什么情况下会命中一级缓存

        相同的 sql 和 参数
        必须是在一个会话 Session当中
        必须是执行 相同的方法
        必须是相同的 namespace (同一个命名空间 -> 同一个mapper文件)
        不能够在查询之前执行 clearCache
        中间不能执行 任何 update ,delete ,insert (会将SqlSession中的数据全部清空)


2.Mybatis的一级缓存机制详解

        一级缓存是SqlSession级别的缓存。我们都知道在操作数据库时需要构造 sqlSession对象,而在sqlSession对象中有一个数据结构(HashMap)用于存储缓存数据。

        一级缓存是SqlSession级别的缓存。在操作数据库时需要构造 sqlSession对象,在对象中有一个(内存区域)数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的。

        一级缓存的作用域是同一个SqlSession,在同一个sqlSession中两次执行相同的sql语句,第一次执行完毕会将数据库中查询的数据写到缓存(内存),第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率。当一个sqlSession结束后该sqlSession中的一级缓存也就不存在了。Mybatis默认开启一级缓存。
        一级缓存只是相对于同一个SqlSession而言。所以在参数和SQL完全一样的情况下,我们使用同一个SqlSession对象调用一个Mapper方法,往往只执行一次SQL,因为使用SelSession第一次查询后,MyBatis会将其放在缓存中,以后再查询的时候,如果没有声明需要刷新,并且缓存没有超时的情况下,SqlSession都会取出当前缓存的数据,而不会再次发送SQL到数据库。

二级缓存

    什么是二级缓存?

        也称为全局缓存,是mapper级别的缓存,是针对一个表的查结果的存储,可以共享给所有针对这张表的查询的用户。

        在同一个namespace下的mapper文件中,执行相同的查询SQL,第一次会去查询数据库,并写到缓存中;第二次直接从缓存中取。当执行SQL时两次查询中间发生了增删改操作,则二级缓存清空。
        使用二级缓存时,由于二级缓存的数据不一定都是存储到内存中,它的存储介质多种多样,所以需要给缓存的对象执行序列化(让pojo对象实现Serializable接口)

    二级缓存步骤:
① 全局配置文件中开启二级缓存
② 需要使用二级缓存的映射文件处使用cache配置缓存
③ 注意:POJO需要实现Serializable接口

   二级缓存相关属性

eviction=“FIFO”:缓存回收策略:
LRU – 最近最少使用的:移除最长时间不被使用的对象。
FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
默认的是 LRU
flushInterval:刷新间隔,单位毫秒
默认情况是不设置,也就是没有刷新间隔,缓存仅仅调用语句时刷新

size:引用数目,正整数
代表缓存最多可以存储多少个对象,太大容易导致内存溢出

readOnly:只读,true/false
true:只读缓存;会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。
false:读写缓存;会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是 false。

运行结果:

注意:二级缓存在sqlsession关闭或者提交之后才会生效
很明显只执行了一次SQL语句

 再来说说下面框起来的这句话,就是缓存中的命中率的意思。

第一次执行语句的时候没有匹配到有缓存的语句,所以命中率为0
第二次是已经有了一个缓因为存了,第二次查的时候就是1/2的命中率,这个0.5就是第二次查询在缓存里面有两条了(因为包括本身查询的这条语句),如果再次进行第三查询的话命中率就是2/3了 

缓存的相关属性设置

  1. 全局setting的cacheEnable:配置二级缓存的开关,一级缓存一直是打开的。
  2. select标签的useCache属性:配置这个select是否使用二级缓存。一级缓存一直是使用的(在全局开启二级缓存的情况下某个select标签关闭二级缓存)
  3. sql标签的flushCache属性:增删改默认flushCache=true。sql执行以后,会同时清空一级和二级缓存。查询默认 flushCache=false。
  4. sqlSession.clearCache():只是用来清除一级缓存。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值