MyBatis一级缓存和二级缓存

news/2024/11/15 6:55:10/

缓存的作用

在 Web 系统中,最重要的操作就是查询数据库中的数据。但是有些时候查询数据的频率非常高,这是很耗费数据库资源的,往往会导致数据库查询效率极低,影响客户的操作体验。于是可以将一些变动不大且访问频率高的数据,放置在一个缓存容器中,用户下一次查询时就从缓存容器中获取结果。

MyBatis 的缓存结构

MyBatis 系统中默认定义了两级缓存:一级缓存和二级缓存:

MyBatis 一级缓存是一个 SqlSession 级别,Sqlsession 只能访问自己的一级缓存的数据。

二级缓存是跨 sqlSession,是 mapper 级别的缓存,对于 mapper 级别的缓存不同的 sqlsession 是可以共享的。

MyBatis 默认开启一级缓存,同时为了增强扩展性,MyBatis 定义了缓存接口 Cache,可以通过 Cache 自定义二级缓存。

一级缓存

MyBatis 一级缓存是一个 SqlSession 级别的缓存,缓存的执行遵循下方的规则:

  1. 映射语句文件中的所有 select 语句的结果将会被缓存。

  2. 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。

  3. 缓存默认会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。

  4. 缓存不会定时进行刷新(也就是说,没有刷新间隔)。

  5. 缓存会保存列表或对象(无论查询方法返回哪种)的 1024 个引用。

  6. 缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。

接下来通过代码模拟一级缓存的执行,用的代码是最简单的一个用户类,首先第一步在 mybatis-config 中开启 log 日志:

<settings><setting name="logImpl" value="STDOUT_LOGGING"/>
</settings>

编写测试代码,在同样的查询条件下查询第二次:

public class CacheTest1 {public static void main(String[] args) {// 获取SqlSessionSqlSession sqlSession = MyBatisUtils.getSqlSession();// 执行SqlUserMapper mapper = sqlSession.getMapper(UserMapper.class);User user=mapper.getUserById(1);System.out.println(user);//同样的条件查询第二次User user2=mapper.getUserById(1);System.out.println(user2);sqlSession.close();}
}

在这里插入图片描述
首先这段代码是在一个 SqlSession 下,因此默认开启了一级缓存,在结果中可以看到,第一次查询走的是数据库,第二次就不需要再查数据库了。满足第一条规则:

映射语句文件中的所有 select 语句的结果将会被缓存。

修改条件,在查询第二次之前先往表里插入一条数据:

public class CacheTest2 {public static void main(String[] args) {// 获取SqlSessionSqlSession sqlSession = MyBatisUtils.getSqlSession();// 执行SqlUserMapper mapper = sqlSession.getMapper(UserMapper.class);User user=mapper.getUserById(1);System.out.println(user);//第二次查询前先插入一条数据User user1=new User(5,"java");mapper.insertUser(user1);//同样的条件查询第二次User user2=mapper.getUserById(2);System.out.println(user2);sqlSession.close();}
}

在这里插入图片描述

在第一次查询之后插入了一条数据,第二次同样条件查询时没有走缓存,再次查表,符合规则:

映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。

二级缓存

二级缓存的作用域比一级缓存要更大,二级缓存是 mapper 级别的缓存,你也可以理解为他是一个 namespace 内的缓存。

开启二级缓存需要几个步骤:

1、 MyBatis 中开启缓存需要首先在设置中开启 cacheEnabled

<settings><setting name="cacheEnabled" value="true"/><setting name="logImpl" value="STDOUT_LOGGING"/>
</settings>

cacheEnabled 默认也是开启状态。

2、 在 mapper.xml 中使用二级缓存

在 UserMapper.xml 文件的 mapper 节点下增加一行 , 这个 mapper 就开启了二级缓存。

MyBatis 缓存要求对应的对象需要实现序列话,因此给 User 对象加上序列化

import java.io.Serializable;public class User implements Serializable{private static final long serialVersionUID = 1L;private int id;private String name;public User(){}public User(int id,String name){this.id=id;this.name=name;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getId() {return id;}public void setId(int id) {this.id = id;}@Overridepublic String toString() {return "id:"+this.id+" name:"+this.name;}
}

编写测试用例,下面这段代码在第一次查询结束后关闭了 SqlSession,接着重新生成一个 SqlSession 执行第二次查询,一级缓存就没有用了,这样的场景下就需要二级缓存。

public class CacheTest3 {public static void main(String[] args) {// 获取SqlSessionSqlSession sqlSession = MyBatisUtils.getSqlSession();// 执行SqlUserMapper mapper = sqlSession.getMapper(UserMapper.class);User user=mapper.getUserById(1);System.out.println(user);// 第一次查询结束后关闭 SqlSessionsqlSession.close();sqlSession = MyBatisUtils.getSqlSession();// 执行Sqlmapper = sqlSession.getMapper(UserMapper.class);//同样的条件查询第二次User user2=mapper.getUserById(1);System.out.println(user2);sqlSession.close();}
}

在这里插入图片描述
使用 useCache 对具体某一个查询设置不适用缓存:

<select id="getUserById" resultMap="UserMap" parameterType="int" useCache="false">select id,name from user where id=#{id};
</select>

cache 标签可以通过配置进行修改:

<cacheeviction="FIFO"flushInterval="60000"size="512"readOnly="false"/>

重点讲一下清除策略(eviction):

  1. LRU – 最近最少使用:移除最长时间不被使用的对象。

  2. FIFO – 先进先出:按对象进入缓存的顺序来移除它们。

  3. SOFT – 软引用:基于垃圾回收器状态和软引用规则移除对象。

  4. WEAK – 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。

这里主要了解 LRU 和 FIFO 即可,默认的清除策略是 LRU。

其他几个属性的配置如下:

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

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

readOnly(只读)属性可以被设置为 true 或 false,默认为 false。只读的缓存会给所有调用者返回缓存对象的相同实例, 因此这些对象不能被修改,这就提供了可观的性能提升。而可读写的缓存会(通过序列化)返回缓存对象的拷贝,速度上会慢一些,但是更安全。


http://www.ppmy.cn/news/7043.html

相关文章

2023年网络安全预测

趋势科技预测2023年&#xff0c;网络犯罪分子仍将活跃并制定新的计划&#xff0c;但一些攻击者仍会坚持使用旧的工具和技术。积极主动的安全和保护对企业来说应该是最重要的。 报告中&#xff0c;根据整个网络安全格局&#xff0c;可分为以下几种&#xff1a; 持续的社会工程…

客快物流大数据项目(一百零一):实时OLAP开发

文章目录 实时OLAP开发 一、实时ETL处理 二、SparkSQL基于DataSourceV2自定义数据源

面试官:ui组件可以自动加载,那么业务组件可以吗?

大厂面试题分享 面试题库 前端面试题库 &#xff08;面试必备&#xff09; 推荐&#xff1a;★★★★★ 地址&#xff1a;前端面试题库 背景 笔者在最近在公司接手了一个老的对内使用的项目&#xff0c;接手后体验了下 发现首屏加载比较慢。分析了下大概的原因是main.js挂…

LeetCode题目笔记——1759. 统计同构子字符串的数目

文章目录题目描述题目难度——中等方法一&#xff1a;数学代码/C代码/Python总结题目描述 给你一个字符串 s &#xff0c;返回 s 中 同构子字符串 的数目。由于答案可能很大&#xff0c;只需返回对 109 7 取余 后的结果。 同构字符串 的定义为&#xff1a;如果一个字符串中的…

Arco 属性

文章目录Arco介绍1. 简介1.1 背景1.2 运行环境1.3 浏览器兼容性2. 设计价值观2.1 清晰2.2 一致2.3 韵律2.4 开放3. 设计原则3.1 及时反馈3.2 贴近现实3.3 系统一致性3.4 防止错误发生3.5 遵从习惯3.6 突出重点3.7 错误帮助3.8 人性化帮助4. 界面总体风格4.1 页面风格4.1.1 主色…

HMS Core 3D流体仿真技术,打造移动端PC级流体动效

移动设备硬件的高速发展&#xff0c;让游戏行业发生翻天覆地的变化&#xff0c;许多酷炫的游戏效果不再局限于电脑端&#xff0c;玩家在移动端就能享受到场景更逼真、画质更清晰、体验更流畅的游戏服务。但由于移动设备算力不足&#xff0c;为了实现真实感的水体效果&#xff0…

求多折线交点

目录 <font color=blue size=4 face="楷体">numpy.isclose<font color=blue size=4 face="楷体">基本思路<font color=blue size=4 face="楷体">代码1<font color=blue size=4 face="楷体">代码2——函数式,…

少儿Python每日一题(2):整数的位数

原题解答 本次的题目如下所示&#xff08;原题出处&#xff1a;蓝桥杯&#xff09;&#xff1a; 【编程实现】 输入一个正整数&#xff0c;输出这个正整数是一个几位数。 输入描述&#xff1a;输入一个正整数 输出描述&#xff1a;输出这个正整数是一个几位数 【样例输入】 12…