mybatis相同的sql查询第二次查不出结果怎么办
mybatis相同的sql查询第二次查不出结果怎么办
这篇文章主要讲解了“mybatis相同的sql查询第二次查不出结果怎么办”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“mybatis相同的sql查询第二次查不出结果怎么办”吧!
相同的sql查询第二次查不出结果问题
问题分析
今天在做公司业务,大致是用户传来订单号(买卡)和手机号对其进行卡绑定,其中如果已经操作过则返回操作后的状态,问题出现在如果是项目启动第一次访问,会正常返回结果,第二次访问时会报找不到卡信息的错误,但是代码是同一个,查询语句也一样,为什么会出现这种情况?找遍百度,未果,自己摸索着寻找问题。
问题探索
首先对查询方法进行了排除,未果;然后寻找代码中问题,发现遍历卡信息(可找到多张卡)时,对于已经操作过的会进行remove()操作,注释掉这行,程序有结果,但是会重新更新卡信息,不是需要的结果;最后发现是对list进行remove()操作时都会出错,于是联想到了MyBatis的缓存。
MyBatis缓存介绍
一级缓存:即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"
<selectid="selectCardInfoByOrderId"resultType="xxx"parameterType="java.util.Map"flushCache="true"></select>
mybatis条件查询容易遇见的错误
在使用mybatis的条件查询时,
一不小心就容易出现这个错误
19-Dec-2017 16:04:38.742 严重 [http-nio-8090-exec-6] org.apache.catalina.core.StandardWrapperValve.invoke Servlet.service() for servlet [SpringMVC] in context with path [] threw exception [Request processing failed; nested exception is org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'endoscopeType' in 'class java.lang.String'] with root causeorg.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'endoscopeType' in 'class java.lang.String'at org.apache.ibatis.reflection.Reflector.getGetInvoker(Reflector.java:380)at org.apache.ibatis.reflection.MetaClass.getGetInvoker(MetaClass.java:170)at org.apache.ibatis.reflection.wrapper.BeanWrapper.getBeanProperty(BeanWrapper.java:152)at org.apache.ibatis.reflection.wrapper.BeanWrapper.get(BeanWrapper.java:48)
这是mapper.xml文件:
<selectid="findByEndoscopeType"resultMap="BaseResultMap"parameterType="java.lang.String">SELECT*FROMendoscope<where><iftest="endoscopeType!=null">endoscope_type=#{endoscopeType,jdbcType=VARCHAR}</if></where></select>
出错的原因是因为加上 <if> 标签时,endoscope属性没有包含在String endoscopeType中
两种解决方法
1.将<if>标签去掉;
2.传入参数放在对象中传进来;
<selectid="findByEndoscopeType"resultMap="BaseResultMap"parameterType="com.iel.endoscope.entity.Endoscope">SELECT*FROMendoscope<where><iftest="endoscopeType!=null">endoscope_type=#{endoscopeType,jdbcType=VARCHAR}</if></where></select>
感谢各位的阅读,以上就是“mybatis相同的sql查询第二次查不出结果怎么办”的内容了,经过本文的学习后,相信大家对mybatis相同的sql查询第二次查不出结果怎么办这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是恰卡编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!
推荐阅读
-
mybatis如何编写一个自定义插件(mybatis plus优点)
mybatisplus优点?Mybatis-Plus是一个Mybatis的增强工具,只是在Mybatis的基础上做了增强却不做改...
-
实用microsoft(sql server 7 教程 怎么配置sqlserver的远程连接)
怎么配置sqlserver的远程连接?SQLServer2008默认是不愿意远程桌面的,如果不是想在本地用SSMS连接远战服务器...
-
如何使用 SQL Server FILESTREAM 存储非结构化数据?
-
安全攻防六:SQL注入,明明设置了强密码,为什么还会被别人登录
-
利用PHP访问MySql数据库以及增删改查实例操作
关于利用PHP访问MySql数据库的逻辑操作以及增删改查实例操作PHP访问MySql数据库˂?php//造连...
-
SQL注入速查表
-
「Web安全」SQL注入的基石
-
每个互联网人才都应该知道的SQL注入
-
MySQL中防止SQL注入
喜欢本文章请关注点赞加转发如何保护数据免受SQL注入攻击?采取措施保护数据免受基于应用程序的攻击,例如SQL注入。千...
-
mybatis中如何防止sql注入和传参