zoukankan      html  css  js  c++  java
  • mybatis一级缓存(session cache)引发的问题

    reference:https://developer.aliyun.com/article/55694

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

  • 相关阅读:
    Andorid之官方导航栏Toobar
    使用ECharts实现数据图表分析
    内存优化之封装九宫格
    xUtils框架的使用详解
    IOS 生成本地验证码
    android之官方导航栏ActionBar(三)之高仿优酷首页
    Android官方导航栏ActionBar(二)—— Action View、Action Provider、Navigation Tabs的详细用法
    IOS 实现界面本地化(国际化)
    Windows7 64位压缩包安装MySQL5.7.9
    Debian 8.2 下安装MySQL5.7.9 Generic Binaries
  • 原文地址:https://www.cnblogs.com/marxtsui/p/14417342.html
Copyright © 2011-2022 走看看