Mybatis——MyBatis 缓存

缓存:cache

缓存的作用:通过减少 IO 的方式,来提高程序的执行效率。

mybatis 的缓存:

将 select 语句的查询结果放到缓存(内存)当中,下一次还是这条 select 语句的话,直接从缓存中取,不再查数据库。一方面是减少了 IO,另一方面不再执行繁琐的查找算法,效率大大提升。

mybatis 缓存包括:

  • 一级缓存:将查询到的数据存储到 SqlSession 中

  • 二级缓存:将查询到的数据存储到 SqlSessionFactory 中

  • 或者集成其它第三方的缓存:比如 EhCache【Java语言开发的】、Memcache【C语言开发的】等

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

一、一级缓存

一级缓存默认是开启的

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

什么情况下不走缓存?

  • 第一种:不同的 SqlSession 对象

  • 第二种:查询条件变化了

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

  • 第一种:第一次查询和第二次查询之间,手动清空了一级缓存

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

二、二级缓存 

二级缓存的范围是 SqlSessionFactory

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

  • <setting name="cacheEnabled" value="true"> 全局性地开启或关闭所有映射器配置文件中已配置的任何缓存。默认就是 true,无需设置

  • 在需要使用二级缓存的 SqlMapper.xml 文件中添加配置:<cache />

  • 使用二级缓存的实体类对象必须是可序列化的,也就是必须实现 java.io.Serializable 接口

  • SqlSession 对象关闭或提交之后,一级缓存中的数据才会被写入到二级缓存当中,此时二级缓存才可用

<cache/>
public class Car implements Serializable {
	//......
}
@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);
}

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

二级缓存的相关配置( <cache/> 的属性):

eviction:指定从缓存中移除某个对象的淘汰算法。默认采用 LRU 策略

  • LRU:Least Recently Used。最近最少使用。优先淘汰在间隔时间内使用频率最低的对象(其实还有一种淘汰算法LFU,最不常用)

  • FIFO:First In First Out。一种先进先出的数据缓存器。先进入二级缓存的对象最先被淘汰

  • SOFT:软引用。淘汰软引用指向的对象。具体算法和 JVM 的垃圾回收算法有关

  • WEAK:弱引用。淘汰弱引用指向的对象。具体算法和 JVM 的垃圾回收算法有关

flushInterval

  • 二级缓存的刷新时间间隔,单位毫秒。如果没有设置,就代表不刷新缓存,只要内存足够大,一直会向二级缓存中缓存数据,除非执行了增删改。

readOnly

  • true:多条相同的 sql 语句执行之后返回的对象是共享的同一个,性能好,但是多线程并发可能会存在安全问题

  • false:多条相同的 sql 语句执行之后返回的对象是副本,调用了 clone 方法,性能一般,但安全

size

  • 设置二级缓存中最多可存储的 Java 对象数量,默认值 1024

三、MyBatis 集成 EhCache

集成 EhCache 是为了代替 mybatis 自带的二级缓存。一级缓存是无法替代的

mybatis 对外提供了接口,也可以集成第三方的缓存组件。比如 EhCache、Memcache 等,都可以

EhCache 是 Java 写的,Memcache 是 C 语言写的,所以 mybatis 集成 EhCache 较为常见

按照以下步骤操作,就可以完成集成:

  • 引入依赖

<!--mybatis集成ehcache的组件-->
<dependency>
  <groupId>org.mybatis.caches</groupId>
  <artifactId>mybatis-ehcache</artifactId>
  <version>1.2.2</version>
</dependency>
  • 在类的根路径下新建 echcache.xml 文件,并提供以下配置信息  
<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
         updateCheck="false">
    <!--磁盘存储:将缓存中暂时不使用的对象,转移到硬盘,类似于Windows系统的虚拟内存-->
    <diskStore path="e:/ehcache"/>
  
    <!--defaultCache:默认的管理策略-->
    <!--eternal:设定缓存的elements是否永远不过期。如果为true,则缓存的数据始终有效,如果为false那么还要根据timeToIdleSeconds,timeToLiveSeconds判断-->
    <!--maxElementsInMemory:在内存中缓存的element的最大数目-->
    <!--overflowToDisk:如果内存中数据超过内存限制,是否要缓存到磁盘上-->
    <!--diskPersistent:是否在磁盘上持久化。指重启jvm后,数据是否有效。默认为false-->
    <!--timeToIdleSeconds:对象空闲时间(单位:秒),指对象在多长时间没有被访问就会失效。只对eternal为false的有效。默认值0,表示一直可以访问-->
    <!--timeToLiveSeconds:对象存活时间(单位:秒),指对象从创建到失效所需要的时间。只对eternal为false的有效。默认值0,表示一直可以访问-->
    <!--memoryStoreEvictionPolicy:缓存的3 种清空策略-->
    <!--FIFO:first in first out (先进先出)-->
    <!--LFU:Less Frequently Used (最少使用).意思是一直以来最少被使用的。缓存的元素有一个hit 属性,hit 值最小的将会被清出缓存-->
    <!--LRU:Least Recently Used(最近最少使用). (ehcache 默认值).缓存的元素有一个时间戳,当缓存容量满了,而又需要腾出地方来缓存新的元素的时候,那么现有缓存元素中时间戳离当前时间最远的元素将被清出缓存-->
    <defaultCache eternal="false" maxElementsInMemory="1000" overflowToDisk="false" diskPersistent="false"
                  timeToIdleSeconds="0" timeToLiveSeconds="600" memoryStoreEvictionPolicy="LRU"/>

</ehcache>
  • 修改 SqlMapper.xml 文件中的 <cache/> 标签,添加 type 属性
<cache type="org.mybatis.caches.ehcache.EhcacheCache"/>
  • 测试 
@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);
}

一  叶  知  秋,奥  妙  玄  心

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

qx_java_1024

祝老板生意兴隆,财源广进!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值