博客
关于我
你凭什么说Spring会导致MyBatis的一级缓存失效
阅读量:342 次
发布时间:2019-03-04

本文共 1985 字,大约阅读时间需要 6 分钟。

Spring和MyBatis集成后,一级缓存失效的原因及解决方案

近年来,关于Spring和MyBatis集成后导致MyBatis的一级缓存失效的说法屡见不鲜。这一现象确实引起了开发者的诸多困惑和不满。为了弄清楚真相,我决定深入研究这一问题。

一、初步分析

首先,我编写了一个简单的测试代码,验证一级缓存是否正常工作。代码如下:

AnnotationConfigApplicationContext annotationConfigApplicationContext;@Beforepublic void init() {    annotationConfigApplicationContext = new AnnotationConfigApplicationContext(AppConfig.class);}@Testpublic void selectTest() {    TestMapper bean = annotationConfigApplicationContext.getBean(TestMapper.class);    List
users = bean.selectUser("周六"); System.out.println(users); List
users1 = bean.selectUser("周六"); System.out.println(users == users1);}

根据MyBatis的一级缓存规则,理论上这两次查询应命中缓存。但测试结果显示,两次查询都直接从数据库查询,缓存未命中。这表明Spring确实导致了MyBatis的一级缓存失效。

二、源码分析

为了找出原因,我决定查看源码。发现MyBatis在Spring集成后,SqlSession被替换成了SqlSessionTemplate。SqlSessionTemplate继承了SqlSession接口,并使用JDK动态代理实现。动态代理的InvocationHandler在处理每个方法调用时,获取了不同的SqlSession。这意味着每次查询使用的是不同的SqlSession,自然无法命中一级缓存。

进一步分析发现,SqlSessionTemplate的selectList方法调用了sqlSessionProxy的selectList,而sqlSessionProxy又是一个动态代理。每次调用都会创建新的SqlSession,这与MyBatis的一级缓存规则不符。

三、解决方案

为了验证这一点,我尝试在代码中添加事务管理。通过获取DataSourceTransactionManager并启动一个事务,确保两次查询使用同一个SqlSession。修改后的代码如下:

@Testpublic void selectTest() {    TestMapper bean = annotationConfigApplicationContext.getBean(TestMapper.class);    DataSourceTransactionManager dataSourceTransactionManager = annotationConfigApplicationContext.getBean(DataSourceTransactionManager.class);    TransactionStatus transaction = dataSourceTransactionManager.getTransaction(new DefaultTransactionDefinition());    List
users = bean.selectUser("周六"); System.out.println(users); List
users1 = bean.selectUser("周六"); System.out.println(users == users1); dataSourceTransactionManager.commit();}

此时,两次查询使用了同一个SqlSession,一级缓存成功命中。

四、总结

Spring和MyBatis集成后,一级缓存失效的原因是SqlSession被动态代理替换,导致每次查询使用不同的SqlSession。通过启用事务管理,确保使用同一个SqlSession,从而解决了问题。

这次调试不仅让我对Spring整合MyBatis有了更深入的理解,也让我对动态代理有了更深刻的认识。后续我将继续分享我的学习心得。

转载地址:http://tqye.baihongyu.com/

你可能感兴趣的文章
PGSQL安装PostGIS扩展模块
查看>>
pg数据库中两个字段相除
查看>>
PhalApi:[1.23] 请求和响应:GET和POST两者皆可得及超越JSON格式返回
查看>>
Phalcon环境搭建与项目开发
查看>>
Phantom.js维护者退出,项目的未来成疑
查看>>
Pharmaceutical的同学们都看过来,关于补码运算的复习相关内容
查看>>
Phaser性能测试加强版
查看>>
phoenix 开发API系列(一)创建简单的http api
查看>>
Phoenix 查看表信息及修改元数据
查看>>
phoenixframework集成了所有自动化测试的思想的平台。mark一下。
查看>>
phoenix_执行sql报错_Error: ERROR 504 (42703): Undefined column. columnName=(state=4270_大数据工作笔记0181
查看>>
phoenix启动失败_The history file `/root/.sqlline/history` may be an older history---记录024_大数据工作笔记0184
查看>>
Phoenix基础命令_视图映射和表映射_数字存储问题---大数据之Hbase工作笔记0036
查看>>
phoenix无法连接hbase shell创建表失败_报错_PleaseHoldException: Master is initializing---记录020_大数据工作笔记0180
查看>>
Phoenix简介_安装部署_以及连接使用---大数据之Hbase工作笔记0035
查看>>
phoenix连接hbase报错Can not resolve hadoop120, please check your network_记录026---大数据工作笔记0187
查看>>
PhotoPrism:这款获得35.8K星的AI照片管理神器你值得拥有
查看>>
Photoshop工作笔记001---Photoshop常用快捷键总结
查看>>
photoshop智能参考线
查看>>
Reids配置文件redis.conf中文详解
查看>>