MyBatis缓存配置的完整示例,包含一级缓存、二级缓存、自定义缓存策略等核心场景,并附详细注释和总结表格

以下是MyBatis缓存配置的完整示例,包含一级缓存、二级缓存、自定义缓存策略等核心场景,并附详细注释和总结表格:


1. 一级缓存(默认开启)

java 复制代码
// 使用同一SqlSession执行两次查询,自动命中一级缓存
try (SqlSession session = sqlSessionFactory.openSession()) {
    UserMapper mapper = session.getMapper(UserMapper.class);
    User user1 = mapper.selectById(1);  // 第一次查询(数据库)
    User user2 = mapper.selectById(1);  // 第二次查询(缓存命中)
    System.out.println(user1 == user2); // 输出:true
}

2. 二级缓存配置(XML方式)

xml 复制代码
<!-- UserMapper.xml -->
<mapper namespace="com.example.mapper.UserMapper">
    <!-- 开启二级缓存 -->
    <cache 
        eviction="FIFO"         <!-- 缓存回收策略:先进先出 -->
        flushInterval="60000"   <!-- 缓存刷新间隔:60秒 -->
        size="512"              <!-- 缓存最大条目数 -->
        readOnly="false"/>      <!-- 是否只读(false允许对象修改) -->
    
    <select id="selectById" resultType="User">
        SELECT * FROM user WHERE id = #{id}
    </select>
</mapper>

3. 二级缓存注解配置

java 复制代码
// UserMapper.java
@Mapper
@CacheNamespace(
    implementation = EhcacheCache.class, // 使用第三方缓存(如Ehcache)
    eviction = CacheNamespaceEvictionPolicy.FIFO,
    flushInterval = 60000,
    size = 512,
    readWrite = true
)
public interface UserMapper {
    User selectById(Integer id);
}

4. 缓存对象序列化配置

java 复制代码
// User实体类需要实现Serializable接口
public class User implements Serializable {
    private Integer id;
    private String name;
    private transient Integer age; // transient字段不参与序列化
    // 省略getter/setter
}

5. 缓存手动操作示例

java 复制代码
// 清除指定对象缓存
sqlSession.clearCache(); // 清除一级缓存
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
mapper.selectById(1); 
sqlSession.commit(); // 触发二级缓存更新

// 清除二级缓存
((org.apache.ibatis.session.Configuration)sqlSession.getConfiguration())
    .getCache("com.example.mapper.UserMapper")
    .clear();

关键技术点总结表

场景 配置方式 适用情况 核心注释
一级缓存 默认开启(SqlSession级别) 单次业务操作内多次查询 不需要配置,关闭需调用sqlSession.clearCache()
二级缓存 <cache>标签或@CacheNamespace 跨SqlSession的全局缓存 必须实现Serializable,需在Mapper配置中显式开启
缓存回收策略 eviction="FIFO/LRU/SID" 控制内存占用 LRU(最近最少使用)最常用
缓存过期 flushInterval="60000" 定时刷新缓存 毫秒单位,配合数据库数据变更频率设置
缓存容量 size="512" 控制内存使用 根据业务数据量合理设置
缓存可写性 readOnly="false" 允许修改缓存对象 readOnly为true时返回对象为不可变副本
第三方缓存 implementation=EhcacheCache 需要分布式缓存 需引入对应依赖并配置缓存实现类

6. 核心配置注意事项:

  1. 序列化要求 :被缓存的对象必须实现Serializable接口
  2. 事务边界:SqlSession关闭或提交(commit)会刷新二级缓存
  3. 并发控制readOnly="true"可避免并发修改问题
  4. 缓存失效:数据更新操作(insert/update/delete)会自动清除相关缓存
  5. 命名空间:二级缓存按Mapper命名空间隔离,需确保唯一性

7. 性能验证示例

java 复制代码
// 验证二级缓存跨SqlSession命中
try (SqlSession session1 = sqlSessionFactory.openSession()) {
    UserMapper mapper1 = session1.getMapper(UserMapper.class);
    User user1 = mapper1.selectById(1); // 数据库查询
}

try (SqlSession session2 = sqlSessionFactory.openSession()) {
    UserMapper mapper2 = session2.getMapper(UserMapper.class);
    User user2 = mapper2.selectById(1); // 二级缓存命中
    System.out.println(user1.equals(user2)); // 输出:true(值相等但对象不同)
}

需要进一步扩展场景(如条件缓存、自定义缓存实现)可补充说明。

相关推荐
好想有猫猫1 小时前
【Redis】服务端高并发分布式结构演进之路
数据库·c++·redis·分布式·缓存
xbhog3 小时前
Java大厂面试突击:从Spring Boot自动配置到Kafka分区策略实战解析
spring boot·kafka·mybatis·java面试·分布式架构
山猪打不过家猪4 小时前
(六)RestAPI 毛子(外部导入打卡/游标分页/Refit/Http resilience/批量提交/Quartz后台任务/Hateoas Driven)
网络·缓存
PXM的算法星球5 小时前
【Java后端】MyBatis 与 MyBatis-Plus 如何防止 SQL 注入?从原理到实战
java·sql·mybatis
旧故新长5 小时前
MyBatis 类型处理器(TypeHandler)注册与映射机制:JsonListTypeHandler和JsonListTypeHandler注册时机
java·开发语言·mybatis
李宥小哥6 小时前
Redis01-基础-入门
缓存·中间件
多多*7 小时前
非关系型数据库 八股文 Redis相关 缓存雪崩 击穿 穿透
java·开发语言·jvm·数据库·redis·缓存·nosql
伊织code9 小时前
cached-property - 类属性缓存装饰器
python·缓存·cache·装饰器·ttl·property·cached-property
李宥小哥10 小时前
Redis03-基础-C#客户端
开发语言·缓存·中间件·c#