美文网首页
Mybatis缓存问题

Mybatis缓存问题

作者: yxuiang | 来源:发表于2018-08-08 19:32 被阅读0次
    • Mybatis在查询时会采用缓存机制,分为一级缓存和二级缓存,一级缓存默认就会开启,二级缓存需要配置才可以使用。
    • 一级缓存基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 Cache 就将清空。
    • 二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap存储,不同在于其存储作用域为 Mapper(Namespace),并且可自定义存储源,如 Ehcache、Hazelcast等。
    • 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了 C/U/D 操作后,默认该作用域下所有 select 中的缓存将被clear。
    • 在缓存的作用域中,如果调用的方法的参数一样,就会使用缓存中查询的结果,而不会再次查询数据库。

    问题:

        public void updateXModel(XAbout xAbout){
            XModel xModelAgo=xModelMapper.selectByPrimaryKey(xAbout.getXModelId());
            if(xModelAgo != null && !xModel.getPhone.equals(xAbout.getPhone)){
                xModelAgo.setPhone(xAbout.getPhone);   //问题做法
                updatePhone(xModelAgo);
                // updatePhone(new XModel(xAbout.getXModelId(),xAbout.getPhone,null,null,null));  // 正确做法
            }
        }
        public void updatePhone(XModel xModel){
            XModel xModelAgo=xModelMapper.selectByPrimaryKey(xModel.getId());
            if(xModelAgo != null && xModelAgo.getPhone.equals(xModel.getPhone)){
                //奇葩的是竟然相等。。。
            }
        }
    

    定位问题:

    mybatis 在查询数据库,返回一个结果,当修改当前结果后,再次执行这个查询方法,返回的是修改后的结果。

    在遇到比较复杂的业务逻辑的时候,比如需要迭代一个方法,获取一个树形结构的数据,此时由于方法会被多次调用,对数据库就会进行多次相同的查询,从第二次调用方法之后就会使用一级缓存。如果你在方法中对查询出来的结果对象直接进行了修改,比如修改了某个属性的值,在下一次调用方法是进行这个查询时,返回的结果是你修改过的结果对象,而不再是从数据库中查询出来的值了,这样可能就和预期的处理不同了,容易造成逻辑错误。因此,应该避免直接操作数据库查询获取的对象。

    解决办法

    当作为参数传入另一个方法做逻辑处理,尽量避免直接操作数据库查询获取的对象;
    应该 重新 new XModel(null,null,null,null,null)  ok!
    

    总结:

    • 这样寡淡贫瘠的一生,遇到你却轰轰烈烈到不像自己。

    相关文章

      网友评论

          本文标题:Mybatis缓存问题

          本文链接:https://www.haomeiwen.com/subject/inoxbftx.html