MyBatis学习笔记之缓存

文章目录

缓存:cache

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

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

不必担心查询到的数据还是原来的数据而不是实时的数据,如果这点小问题还得我们自己做这个框架就是失败的


mybatis缓存包括:

  • 一级缓存:将查询到的数据存储到SqlSession中。
  • 二级缓存:将查询到的数据存储到SqlSessionFactory中。
  • 或者集成其他第三方的缓存:比如EhCache【Java语言开发的】、Memcache【C语言开发的】等.

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


拓展:

常见的缓存技术手段

  • 字符串常量池
  • 整数型常量池
  • 线程池
  • 连接池
  • and so on

一级缓存

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

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

java 复制代码
@Test
public void testSelectById(){
	SqlSession sqlSession = SqlSessionUtil.openSession();
	
	CarMapper mapper1 = sqlSession.getMapper(CarMapper.class);
	Car car1 = mapper1.selectById(164L);
	System.out.println(car1);

	CarMapper mapper2 = sqlSession.getMapper(CarMapper.class);
	Car car2 = mapper2.selectById(164L);
	System.out.println(car2);

	sqlSession.close();
}

通过输出日志可以看出,只执行了一条SQL语句

一级缓存失效

第一次DQL和第二次DQL之间做了任意以下两件事情中的一件都会让一级缓存失效

  • 第一种:第一次查询和第二次查询之间,手动清空了一级缓存
java 复制代码
sqlSession.clearCache();
  • 第二种:第一次查询和第二次查询之间,进行了增删改操作。(这个增删改和哪张表没有关系,只要有insert、delete、update操作,一级缓存就失效)

二级缓存

二级缓存的范围是SqlSessionFactory


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

  1. <setting name="cacheEnabled" value="true>"全局性地开启或关闭所有映射器配置文件中已配置地任何缓存,默认就是true,无需配置。
  2. 在需要使用二级缓存的SqlMapper.xml文件中添加配置:<cache/>
  3. 使用二级缓存的实体类对象必须是可序化的,也就是必须实现java.io.Serializable接口
  4. SqlSession对象关闭或提交之后一级缓存中的数据才会被写入到二级缓存当中。此时二级缓存才可用。
java 复制代码
@Test
public void testSelectById2() throws Exception{
	// 这里只有一个SqlSessionFactory对象,二级缓存对应的就是SqlSessionFactory
	SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));
	SqlSession sqlSession1 = sqlSessionFactory.openSession();
	SqlSession sqlSession2 = sqlSessionFactory.openSession();
	CarMapper mapper1 = sqlSession1.getMapper(CarMapper.class);
	CarMapper mapper2 = sqlSession1.getMapper(CarMapper.class);
		
	//这行代码执行结束之后,实际上数据是缓存到一级缓存当中了。(sqlSession1是一级缓存。)
	Car car1 = mapper1.selectById(10086L);
	System.out.println(car1);
	
	//如果这里不关闭SqlSession1对象的话,二级缓存中还是没有数据的。
	
	//这行代码执行结束之后,实际上数据是缓存到一级缓存当中了。(sqlSession2是一级缓存。)
	Car car2 = mapper2.selectById(10086L);
	System.out.println(car2);
	
	//程序执行到这里的时候,会将sqlSession1这个一级缓存中的数据写入到二级缓存当中
	sqlSession1.close();
	//程序执行到这里的时候,会将sqlSession2这个一级缓存中的数据写入到二级缓存当中
	sqlSession1.close();
}

如果将sqlSession1的close放到car2定义的前面

二级缓存失效

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

二级缓存相关配置

  1. eviction(驱逐):指定从缓存中溢出某个对象的淘汰算法。默认采用LRU策略。
    a. LRU:Least Recently Used。最近最少使用。优先淘汰再建个时间内使用频率最低的对象。(其实还有一种淘汰算法LFU,最不常用)
    b. FIFO:First In First Out。一种先进先出的数据缓存器。先进入二级缓存的对象最先被淘汰
    c. SOFT:软引用。淘汰软引用指向的对象。具体算法和JVM的垃圾回收算法有关。
    d. WEAK:弱引用。淘汰引用指向的对象。具体的算法和JVM的垃圾回收算法有关。
  2. flushInterval:
    a. 二级缓存的刷新时间间隔。单位毫秒。如果没有设置,就代表不刷新缓存,只要内存足够大,一直会向二级缓存中缓存数据。除非执行了增删改。
  3. readOnly:
    a. true:多条相同的sql语句执行之后返回的对象是共享的同一个。性能好,但是多线程并发可能会存在安全问题。
    b. false:多条相同的sql语句之后返回的对象是副本,调用了close方法,性能一般,但安全。
  4. size
    a. 设置二级缓存中最多可存储的java数量,默认值1024

MyBatis集成EhCache

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

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

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


下面是如何配置

第一步:引入mybatis整合ehcache的依赖

xml 复制代码
<!-- pom.xml -->
        <dependency>
            <groupId>org.mybatis.caches</groupId>
            <artifactId>mybatis-ehcache</artifactId>
            <version>1.2.2</version>
        </dependency>

第二步:在类的根路径下新建ehcache.xml文件,并提供以下配置信息

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=""/>

    <!--defaultCache:默认的管理策略-->
    <!--eternal:设定缓存的elements是否永远不过期。如果为true,则缓存的数据始终有效;如果为false那么还要根据timeToIdleSeconds,timeToLiveSeconds判断-->
    <!--maxElementsInMemory:在内存中缓存的element的最大数目-->
    <!--overflowToDisk:如果内存中数据超过内存限制,是否缓存到磁盘上-->
    <!--diskPersistent:是否在磁盘上持久化,指重启JVM后,数据是否有效,默认为false-->
    <!--timeToIdleSeconds:对象空闲时间(单位:秒),指对象在多长时间没有被访问就会失效。只对eternal为false的有效,默认值为0,表示一直可以访问-->
    <!--timeToLiveSeconds:对象存活时间(单位:秒),指对象从创建到失败所需要的时间。只对eternal为false的有效。默认值为0,表示一直可以访问-->
    <!--memoryStoreEvictionPolicy:缓存的三种清空策略-->
    <!--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属性。

xml 复制代码
<cache type="org.mybatis.caches.ehcache.EhcacheCache"/>

然后就能运行了

相关推荐
行然梦实15 分钟前
学习日记_20241110_聚类方法(K-Means)
学习·kmeans·聚类
马船长21 分钟前
制作图片木马
学习
秀儿还能再秀32 分钟前
机器学习——简单线性回归、逻辑回归
笔记·python·学习·机器学习
WCF向光而行37 分钟前
Getting accurate time estimates from your tea(从您的团队获得准确的时间估计)
笔记·学习
wang09071 小时前
工作和学习遇到的技术问题
学习
Li_0304062 小时前
Java第十四天(实训学习整理资料(十三)Java网络编程)
java·网络·笔记·学习·计算机网络
心怀梦想的咸鱼2 小时前
ue5 蓝图学习(一)结构体的使用
学习·ue5
kali-Myon3 小时前
ctfshow-web入门-SSTI(web361-web368)上
前端·python·学习·安全·web安全·web
啤酒泡泡_Lyla3 小时前
现代无线通信接收机架构:超外差、零中频与低中频的比较分析
笔记·信息与通信
龙中舞王3 小时前
Unity学习笔记(4):人物和基本组件
笔记·学习·unity