mybatis一级缓存(session cache)引发的问题

简介: 问题回顾 ------------- 最近项目功能单元测试中,出现了一个奇怪的bug。远程调试发现,程序进行了2次相同的查询,返回了实体类(ClassA)的2个对象:classAInstance1和classAInstance2,当修改classAInstance1.property1时,竟然classAInstance2.property1也被改了!!! 很快发现classAInstan

问题回顾

最近项目功能单元测试中,出现了一个奇怪的bug。远程调试发现,程序进行了2次相同的查询,返回了实体类(ClassA)的2个对象:classAInstance1和classAInstance2,当修改classAInstance1.property1时,竟然classAInstance2.property1也被改了!!! 很快发现classAInstance1和classAInstance2地址是相同的,它们是同一个内存对象!

原因分析

经调试发现,mybatis返回的实体类的内存地址是相同的!于是猜测是mybatis缓存的原因,于是进行了下面的测试验证

测试验证

经过单元测试验证,不开启事务的情况下,多次相同的查询,返回对象地址不相等, 代码略。
经过单元测试验证,在一个事务内,多次相同的查询,返回对象地址相等, 代码如下:

    
    @Resource
    private MybatisSessionCacheTestService mybatisSessionCacheTestService;

    @Test
    public void test_mybatis_sql_session_cache(){
        Long id = 100L;
        ClassA classAInstance1 = mybatisSessionCacheTestService.queryById(id);
        ClassA classAInstance2 = mybatisSessionCacheTestService.queryById(id);
        //assert mybatis cache is on
        Assert.assertTrue(classAInstance1 == classAInstance2);
    }
    @Service
    public class MybatisSessionCacheTestService {
        @Resource
        private ClassADAO classDAO;

        @Transactional //spring 事务注解
        public ClassA queryById(Long id){
            return classADAO.queryById(id);
        }
    }

解决方案

  • 1.把查询提前到事务之前(之外),这样只解决了个别问题,解决并不彻底。
  • 2.在mybatis的mapper xml里配置每次清空缓存flushCache:
<select id="selectById" resultType="ClassA" flushCache="true">
...
</select>

附录:mybatis缓存介绍

一级缓存

即session缓存,作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 Cache 就将清空,默认开启。

注意 集成spring(使用mybatis-spring)时:

  • 每次查询spring会重新创建SqlSession,所以一级缓存是不生效的。
  • 而当开启事务时,spring会使用同一个SqlSession做查询,所以这个情况下一级缓存是生效的

二级缓存

即全局缓存,其作用域为 Mapper(Namespace),默认关闭。

其他参考:

目录
相关文章
|
3月前
|
缓存 Java 数据库连接
MyBatis的缓存
MyBatis的缓存
|
4月前
|
存储 缓存 Java
干翻Mybatis源码系列之第八篇:Mybatis二级缓存的创建和存储
干翻Mybatis源码系列之第八篇:Mybatis二级缓存的创建和存储
|
1月前
|
存储 缓存 Java
什么!?实战项目竟然撞到阿里面试的原题!???关于MyBatis Plus的缓存机制
什么!?实战项目竟然撞到阿里面试的原题!???关于MyBatis Plus的缓存机制
|
1月前
|
缓存 Java 数据库连接
mybatis 数据库缓存的原理
MyBatis 是一个流行的 Java 持久层框架,它封装了 JDBC,使数据库交互变得更简单、直观。MyBatis 支持两级缓存:一级缓存(Local Cache)和二级缓存(Global Cache),通过这两级缓存可以有效地减少数据库的访问次数,提高应用性能。
282 1
|
1月前
|
存储 缓存 Java
【MyBaits】4、延迟加载、MyBatis 的缓存
【MyBaits】4、延迟加载、MyBatis 的缓存
22 0
|
2月前
|
SQL 缓存 Java
mybatis缓存详解
mybatis缓存详解
22 0
|
3月前
|
缓存 Java 数据库连接
mybatis的缓存内容(下)
mybatis的缓存内容
28 0
|
3月前
|
SQL 缓存 Java
mybatis的缓存内容(上)
mybatis的缓存内容
31 0
|
3月前
|
缓存 Java 数据库连接
|
3月前
|
缓存 Java 数据库连接
MyBatis支持的缓存刷新模式
MyBatis支持的缓存刷新模式
217 1