MyBatis 缓存机制详解

server/2024/10/18 8:19:58/

MyBatis 是一款优秀的持久层框架,通过配置文件或注解方式简化了数据库操作。为了提高性能,MyBatis 提供了两级缓存机制:一级缓存和二级缓存。本文将详细介绍 MyBatis缓存机制,帮助你理解和应用这一强大功能,提升应用程序的性能。

1. 缓存的基本概念

缓存(Cache)是指存储数据的临时存储区域,以便快速访问。缓存可以显著减少数据库查询的次数,从而提高应用程序的性能。MyBatis 提供了两级缓存机制:

  • 一级缓存(本地缓存):作用于 SQL 会话范围内(SqlSession),默认启用,不能跨会话共享。
  • 二级缓存(全局缓存):作用于映射器(Mapper)范围内,多个会话可以共享,默认关闭,需要手动配置。

2. 一级缓存

一级缓存MyBatis 默认开启的,会话级别的缓存。只要在同一个 SqlSession 中,查询的数据会被缓存,下次查询相同的数据时,会直接从缓存中获取,而不会再执行 SQL 语句。

2.1 一级缓存示例

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;public class MyBatisCacheExample {public static void main(String[] args) {SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);try (SqlSession session = sqlSessionFactory.openSession()) {UserMapper mapper = session.getMapper(UserMapper.class);// 第一次查询,执行 SQL 语句User user1 = mapper.getUserById(1);System.out.println(user1);// 第二次查询,同一个会话,使用缓存User user2 = mapper.getUserById(1);System.out.println(user2);// 检查是否使用了缓存System.out.println(user1 == user2); // true}}
}

在上述代码中,第二次查询时,MyBatis 会直接从一级缓存中获取数据,而不会再执行 SQL 语句。

2.2 一级缓存失效情况

在以下情况下,一级缓存会失效:

  • 执行 INSERTUPDATEDELETE 操作
  • 显式调用 SqlSessionclearCache 方法
  • 会话关闭

3. 二级缓存

二级缓存是作用于映射器(Mapper)级别的缓存,不同 SqlSession 之间可以共享。默认情况下,二级缓存是关闭的,需要手动配置。

3.1 启用二级缓存

首先,需要在 MyBatis 的全局配置文件(mybatis-config.xml)中启用二级缓存

<configuration><settings><setting name="cacheEnabled" value="true"/></settings>
</configuration>

然后,在映射器 XML 文件(如 UserMapper.xml)中配置二级缓存

<mapper namespace="com.example.mapper.UserMapper"><cache/><select id="getUserById" resultType="com.example.model.User">SELECT id, name, email FROM users WHERE id = #{id}</select>
</mapper>

3.2 二级缓存示例

public class MyBatisCacheExample {public static void main(String[] args) {SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);try (SqlSession session1 = sqlSessionFactory.openSession()) {UserMapper mapper1 = session1.getMapper(UserMapper.class);User user1 = mapper1.getUserById(1);System.out.println(user1);}try (SqlSession session2 = sqlSessionFactory.openSession()) {UserMapper mapper2 = session2.getMapper(UserMapper.class);User user2 = mapper2.getUserById(1);System.out.println(user2);}}
}

在上述代码中,第二次查询时,尽管是不同的 SqlSessionMyBatis 仍会从二级缓存中获取数据。

3.3 二级缓存配置选项

二级缓存可以配置多种属性,以满足不同的需求:

<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
  • eviction缓存回收策略(默认是 LRU,支持 FIFO、SOFT、WEAK)
  • flushInterval:刷新间隔(以毫秒为单位),默认不设置(不刷新)
  • size缓存的引用数目,默认值是 1024
  • readOnly:只读属性,默认值是 false(可读可写)

3.4 自定义缓存

MyBatis 允许用户自定义缓存实现,只需要实现 org.apache.ibatis.cache.Cache 接口,并在映射器中配置即可:

public class CustomCache implements Cache {private final String id;public CustomCache(String id) {this.id = id;}@Overridepublic String getId() {return this.id;}@Overridepublic void putObject(Object key, Object value) {// 自定义缓存存储逻辑}@Overridepublic Object getObject(Object key) {// 自定义缓存获取逻辑return null;}@Overridepublic Object removeObject(Object key) {// 自定义缓存移除逻辑return null;}@Overridepublic void clear() {// 自定义缓存清除逻辑}@Overridepublic int getSize() {// 自定义缓存大小return 0;}
}

在映射器 XML 文件中配置自定义缓存

<cache type="com.example.cache.CustomCache"/>

4. 缓存策略和注意事项

缓存可以显著提高性能,但也可能引发数据一致性问题。以下是一些缓存策略和注意事项:

  • 对于频繁变动的数据,不适合使用缓存
  • 对于读多写少的数据,可以考虑使用二级缓存
  • 在使用缓存时,要注意缓存的失效策略和数据的一致性。

5. 总结

MyBatis 提供了灵活的一级和二级缓存机制,可以显著提高数据库访问性能。一级缓存默认启用,作用于会话范围内;二级缓存需要手动配置,作用于映射器范围内。通过合理配置缓存,可以减少数据库查询次数,提高应用程序的响应速度。

本文详细介绍了 MyBatis缓存机制、使用方法及注意事项,希望能帮助你更好地掌握 MyBatis 缓存的使用,提升开发效率和应用性能。

进一步学习 MyBatis 缓存机制,可以参考以下资源:

希望本文对你有所帮助,祝你在使用 MyBatis 时取得更好的成果。


http://www.ppmy.cn/server/49786.html

相关文章

面向对象的三大特性与类图

1. 面向对象编程的三大特点 Object-oriented programming (OOP) is a paradigm centered around the concept of objects, which can contain data and code to manipulate that data. The three major characteristics of object-oriented programming are encapsulation, in…

第十一章:接口

接口 文章目录 接口一、简介1.1 接口是什么1.2 接口的作用1.3 接口的开发与调用1.4 接口的组成 二、RESTful API三、json-server四、接口测试工具五、接口的创建 一、简介 1.1 接口是什么 接口是前后端通信的桥梁 简单理解&#xff1a;一个接口就是 服务中的一个路由规则&am…

数据库第一次实验报告

1. 使用 SQL 语句创建数据库 studentsdb。 2. 使用 SQL 语句选择 studentsdb 为当前使用数据库。 3. 使用 SQL 语句在 studentsdb 数据库创建数据表 student_info、curriculum、grade 4. 使用 SQL 语句 INSERT 向 studentsdb 数据库的 student_info、curriculum、grade 表插…

JAVA8 常用Stram处理方法

JAVA8 常用Stram处理方法 排序排序对象集合属性一升序 属性二降序List转MapList分组求和提取字符串根据属性去重分组排序求和 排序 List<MachineOrderResponse.BackRecord> noSList ss.stream().sorted(Comparator.comparing(MachineOrderResponse.BackRecord::getTime)…

分片表属性的查询

select tabinfo.表名,f.partition 分区名,tabinfo.存储空间,tabinfo.段数量,tabinfo.分配总数,tabinfo.已用数量,tabinfo.行长,tabinfo.行数 from (select a.tabname 表名, c.name 存储空间,b.nextns 段数量,b.nptotal 分配总数,b.npused 已用数量,b.rowsize 行长,b.nrows 行数…

Linux C编译器从零开发三

AST语法树 BNF抽象 expr equality equality relational ("" relational | "!" relational)* relational add ("<" add | "<" add | ">" add | ">" add)* add mul ("" …

Oracle 19C 数据库表被误删除的模拟恢复

Oracle 19C 数据库表被误删除的模拟恢复操作 1、模拟创建表用于恢复测试 sqlplus zzh/zzh SQL> create table obj_tb tablespace users as select * from dba_objects; Table created. SQL> select count(*) from obj_tb; COUNT(*) ---------- 72373 2、记录当前…

SGPT论文阅读笔记

这是篇想要用GPT来提取sentence embedding的工作&#xff0c;提出了两个框架&#xff0c;一个是SGPT-BE&#xff0c;一个是SGPT-CE&#xff0c;分别代表了Bi-Encoder setting和Cross-Encoder setting。CE的意思是在做阅读理解任务时&#xff0c;document和query是一起送进去&am…