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

问题回顾

最近项目功能单元测试中,出现了一个奇怪的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),默认关闭。

其他参考:

附注jar包版本:

<dependency>
    <groupId>org.mybatis</groupId>
    <artifactId>mybatis</artifactId>
    <version>3.3.0</version>
</dependency>
<dependency>
    <groupId>org.mybatis</groupId>
    <artifactId>mybatis-spring</artifactId>
    <version>1.3.0</version>
</dependency>
  • 6
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Spring Boot中使用MyBatis时,默认情况下是没有开启二级缓存的。要开启二级缓存,可以通过在全局配置文件(mybatis-config.xml)中进行配置,或者在application.yml文件中进行配置。 二级缓存是指在mapper-namespace级别的缓存,它可以提高查询性能,避免频繁地访问数据库。一级缓存是指在session级别的缓存,它默认是开启的,而且无法关闭。 要开启MyBatis的二级缓存,你可以在全局配置文件(mybatis-config.xml)中添加以下配置: ``` <configuration> <settings> <setting name="cacheEnabled" value="true" /> </settings> </configuration> ``` 或者在application.yml文件中添加以下配置: ``` mybatis: configuration: cache-enabled: true ``` 这样就可以开启MyBatis的二级缓存了。注意,开启二级缓存后,需要在Mapper接口的方法上使用@CacheNamespace注解来启用缓存。同时,还需要确保你的实体类实现了Serializable接口,以便支持缓存的序列化和反序列化操作。 总结起来,Spring Boot默认情况下是没有开启MyBatis二级缓存的,你需要进行相应的配置来启用二级缓存,并在Mapper接口的方法上使用@CacheNamespace注解来启用缓存。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [springboot-mybatis之二级缓存(注解方式)-随手记](https://blog.csdn.net/luning95/article/details/97756669)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [springboot+mybatis+redis 二级缓存问题实例详解](https://download.csdn.net/download/weixin_38618315/12767315)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值