-
问题分析:今天在做公司业务,大致是用户传来订单号(买卡)和手机号对其进行卡绑定,其中如果已经操作过则返回操作后的状态,问题出现在如果是项目启动第一次访问,会正常返回结果,第二次访问时会报找不到卡信息的错误,但是代码是同一个,查询语句也一样,为什么会出现这种情况?找遍百度,未果,自己摸索着寻找问题。
-
问题探索:首先对查询方法进行了排除,未果;然后寻找代码中问题,发现遍历卡信息(可找到多张卡)时,对于已经操作过的会进行remove()操作,注释掉这行,程序有结果,但是会重新更新卡信息,不是需要的结果;最后发现是对list进行remove()操作时都会出错,于是联想到了MyBatis的缓存。
-
MyBatis缓存介绍[1]:
一级缓存:即session缓存,作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 Cache 就将清空,默认开启。注意:集成spring(使用mybatis-spring)时:
-
每次查询spring会重新创建SqlSession,所以一级缓存是不生效的;
-
而当开启事务时,spring会使用同一个SqlSession做查询,所以这个情况下一级缓存是生效的。
二级缓存:即全局缓存,其作用域为 Mapper(Namespace),默认关闭。
-
-
问题原因:上述提到了MyBatis的缓存机制,查看项目配置后发现问题在于第一次查询到结果会放到缓存中,程序对查到的结果list进行了remove操作,所以缓存中的list会发生变化,第二次查询时会从缓存中将操作过的list查找出来(mybatis返回的实体类的内存地址是相同的),故而产生了我们不需要的结果,之前项目中使用的是与Spring集成的,使用的session是SqlSessionTemplate,这里是默认关闭了一级缓存,而今天项目中没有与Spring集成,创建session使用的是SqlsessionFactory的openSession()方法,这里查找时默认会先从缓存中查询,综上,第二次我们查到的只是缓存中的数据[2]。
-
解决方案:既然第二次会从缓存中读数据,不可能修改项目配置关闭一级缓存,所以可以通过刷新缓存来达到我们所需要的目的
方案一:通过SqlSessionUtils.getSqlSession(sqlSessionFactory).clearCache()方法刷新缓存
方案二:在mapper.xml对应的查找语句中添加flushCache="true"
<select id="selectCardInfoByOrderId" resultType="xxx" parameterType="java.util.Map" flushCache="true"></select>
-
总结:此次问题的发生与解决对MyBatis缓存进一步了解,至于提供的两种方案有什么区别还需要继续探索,并且接下来要对SqlSessionTemplate[3]和SqlSession[4]做更深层次的学习,最后感谢提供相关资料的各位博主!
MyBatis同一个查询第一次能查到第二次查不到数据?
最后编辑于 :
?著作权归作者所有,转载或内容合作请联系作者
- 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
- 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
- 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
推荐阅读更多精彩内容
- 11 MyBatis一级缓存实现# 11.1 什么是一级缓存? 为什么使用一级缓存?## 每当我们使用MyBati...
- 前言 主题是Mybatis一级和二级缓存的应用及源码分析。希望在本场chat结束后,能够帮助读者朋友明白以下三点。...