mybatis相同的sql查询第二次查不出结果问题

 更新时间:2022年01月21日 15:22:58   作者:暗香抚动  
这篇文章主要介绍了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"

<select id="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 cause org.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文件:

<select id="findByEndoscopeType" resultMap="BaseResultMap" parameterType="java.lang.String">
    SELECT *
    FROM endoscope
    <where>
        <if test="endoscopeType != null">
            endoscope_type = #{endoscopeType,jdbcType=VARCHAR}
        </if>
    </where>
</select>

出错的原因是因为加上 <if> 标签时,endoscope属性没有包含在String endoscopeType中

两种解决方法

1.将<if>标签去掉;

2.传入参数放在对象中传进来;

<select id="findByEndoscopeType" resultMap="BaseResultMap" parameterType="com.iel.endoscope.entity.Endoscope">
    SELECT *
    FROM endoscope
    <where>
        <if test="endoscopeType != null">
            endoscope_type = #{endoscopeType,jdbcType=VARCHAR}
        </if>
    </where>
</select>

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Java Metrics系统性能监控工具的使用详解

    Java Metrics系统性能监控工具的使用详解

    Metrics是一个Java库,可以对系统进行监控,统计一些系统的性能指标。本文就来和大家详细聊聊这个工具的具体使用,希望对大家有所帮助
    2022-11-11
  • java高并发ScheduledThreadPoolExecutor与Timer区别

    java高并发ScheduledThreadPoolExecutor与Timer区别

    这篇文章主要为大家介绍了java高并发ScheduledThreadPoolExecutor与Timer区别,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10
  • Geotools实现shape文件的写入功能

    Geotools实现shape文件的写入功能

    Geotools作为开源的Java GIS三方库,已经成为GIS服务器端的主流开源库,其功能非常强大,涉及到GIS业务的方方面面,其中就包括GIS数据的读写,今天小编就借助Geotools来实现shape数据的写入,需要的朋友可以参考下
    2023-08-08
  • JAVA如何使用Math类操作数据

    JAVA如何使用Math类操作数据

    这篇文章主要介绍了JAVA如何使用Math类操作数据,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • 解决springboot使用logback日志出现LOG_PATH_IS_UNDEFINED文件夹的问题

    解决springboot使用logback日志出现LOG_PATH_IS_UNDEFINED文件夹的问题

    这篇文章主要介绍了解决springboot使用logback日志出现LOG_PATH_IS_UNDEFINED文件夹的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • RabbitMQ消息队列的目录结构

    RabbitMQ消息队列的目录结构

    这篇文章主要介绍了RabbitMQ消息队列的目录结构,RabbitMQ 属于消息中间件,主要用于组件之间的解耦,消息的发送者无需知道消息使用者的存在,反之亦然,那么用了那么久RabbitMQ,其目录结构是怎样的呢,让我们一起来看一下吧
    2023-08-08
  • Java实现数独小游戏

    Java实现数独小游戏

    这篇文章主要为大家详细介绍了Java实现数独小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-05-05
  • Java里的static import使用小结

    Java里的static import使用小结

    这篇文章主要介绍了Java里的static import使用小结,本文给出了一些使用示例,并总结了一些使用特性,需要的朋友可以参考下
    2015-06-06
  • Java JVM虚拟机运行机制

    Java JVM虚拟机运行机制

    JVM(Java虚拟机)一种用于计算设备的规范,可用不同的方式(软件或硬件)加以实现。接下来通过本文给大家简单介绍Java JVM虚拟机运行机制,感兴趣的朋友一起看看吧
    2017-03-03
  • 解决json串和实体类字段不一致的问题

    解决json串和实体类字段不一致的问题

    这篇文章主要介绍了解决json串和实体类字段不一致的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-03-03

最新评论