【问题处理】—— Mybatis缓存可改导致的异常

现象:数组越界

        List<Integer> dateList = commonMapper.queryDateBefore(count);
        dateList.remove(0);
        return dateList.get(dateList.size() - 1);

报的是数组越界,错误下标是-1;然而我们从日志看commonMapper.queryDateBefore 返回的结果是8条;8条去掉1条,怎么会变成0了呢?

分析:缓存与返回同对象

一开始百思不得其解,翻遍日志,可以看到所有调用commonMapper.queryDateBefore的返回都是8条,而这里的List dateList是个局部变量,也不会有多线程来干扰

甚至看了remove()的源码,当然结果一无所获

直到后面打了断点,发现这个地方会执行多次,突然想到会不会是myBatis缓存问题,我们看到的日志是8没错,但是同一个线程多次执行此Sql,我印象中后续是会直接返回缓存,而不是再查数据库的,那myBatis自然不会再打印查询日志了,所以造成了所有查询的结果都是8条

这其实有点让人意外,因为我一直认为myBatis的缓存是一种备份形式,没想到其缓存和返回值就是同一个对象,对查出来的数据进行修改后,再次查询其原值居然是错的

为了验证这种猜测,找到了缓存部分的源码:org.apache.ibatis.executor.BaseExecutor
myBatis
果然,在查询list的时候,会把该list存储在缓存中再返回,如果我们对返回值进行操作,那实际也就是在操作缓存内容,导致本线程第二次查询时,得到错误的数据

结果:拷贝后修改

在查询出结果后,不直接对原list进行操作,而是对list采用一次部分深拷贝,然后再操作新集合

        List<Integer> dateList = commonMapper.queryDateBefore(count);
        ArrayList<Object> newDateList = new ArrayList<>(dateList);
        newDateList.remove(0);
        return newDateList.get(newDateList.size() - 1);
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

战斧

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值