zoukankan      html  css  js  c++  java
  • Mybatis 为什么不要用二级缓存

    https://www.cnblogs.com/liouwei4083/p/6025929.html

    mybatis 二级缓存不推荐使用

        一 mybatis的缓存使用。

    大体就是首先根据你的sqlid,参数的信息自己算出一个key值,然后你查询的时候,会先把这个key值去缓存中找看有没有value,如果有,直接返回出来,就不查询db了。如果没有,那么查询db,然后将key,value保存到缓存中,以便下次使用。

         1.1mybatis的一级缓存是基于sqlsession为生命周期的

    当你这个session没有了,缓存就没有了,其次当你sql执行!isselect语句的时候,缓存也会被直接全部清理掉以保证数据一致性。
    复制代码
      public int update(MappedStatement ms, Object parameter) throws SQLException {
        ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());
        if (closed) throw new ExecutorException("Executor was closed.");
         //清理缓存
        clearLocalCache();
        return doUpdate(ms, parameter);
      }
    复制代码

         1.2 mybatis的二级缓存是基于application为生命周期的

    范围是按照每个namepace一个缓存来存贮和维护,同一个namespace放到一个缓存对象中,当这个namaspace中执行了!isselect语句的时候,整个namespace中的缓存全部清除掉。
      public int update(MappedStatement ms, Object parameterObject) throws SQLException {
         //清理缓存,并且!isselect语句的flushcache都是默认为true的。
        flushCacheIfRequired(ms);
        return delegate.update(ms, parameterObject);
      }
     
    以下摘抄自网络观点文字  http://blog.csdn.net/isea533/article/details/44566257

         三、Cache使用时的注意事项


    1. 只能在【只有单表操作】的表上使用缓存

    不只是要保证这个表在整个系统中只有单表操作,而且和该表有关的全部操作必须全部在一个namespace下。

    2. 在可以保证查询远远大于insert,update,delete操作的情况下使用缓存

    这一点不需要多说,所有人都应该清楚。记住,这一点需要保证在1的前提下才可以!

    四、避免使用二级缓存


    可能会有很多人不理解这里,二级缓存带来的好处远远比不上他所隐藏的危害。

    缓存是以namespace为单位的,不同namespace下的操作互不影响。

    insert,update,delete操作会清空所在namespace下的全部缓存。

    通常使用MyBatis Generator生成的代码中,都是各个表独立的,每个表都有自己的namespace。

    为什么避免使用二级缓存

    在符合【Cache使用时的注意事项】的要求时,并没有什么危害。

    其他情况就会有很多危害了。

    针对一个表的某些操作不在他独立的namespace下进行。

    例如在UserMapper.xml中有大多数针对user表的操作。但是在一个XXXMapper.xml中,还有针对user单表的操作。

    这会导致user在两个命名空间下的数据不一致。如果在UserMapper.xml中做了刷新缓存的操作,在XXXMapper.xml中缓存仍然有效,如果有针对user的单表查询,使用缓存的结果可能会不正确。

    更危险的情况是在XXXMapper.xml做了insert,update,delete操作时,会导致UserMapper.xml中的各种操作充满未知和风险。

    有关这样单表的操作可能不常见。但是你也许想到了一种常见的情况。

    多表操作一定不能使用缓存

    为什么不能?

    首先不管多表操作写到那个namespace下,都会存在某个表不在这个namespace下的情况。

    例如两个表:role和user_role,如果我想查询出某个用户的全部角色role,就一定会涉及到多表的操作。

    <select id="selectUserRoles" resultType="UserRoleVO">
        select * from user_role a,role b where a.roleid = b.roleid and a.userid = #{userid}
    </select>
    像上面这个查询,你会写到那个xml中呢??

    不管是写到RoleMapper.xml还是UserRoleMapper.xml,或者是一个独立的XxxMapper.xml中。如果使用了二级缓存,都会导致上面这个查询结果可能不正确。

    如果你正好修改了这个用户的角色,上面这个查询使用缓存的时候结果就是错的。

    这点应该很容易理解。

    在我看来,就以MyBatis目前的缓存方式来看是无解的。多表操作根本不能缓存。

    如果你让他们都使用同一个namespace(通过<cache-ref>)来避免脏数据,那就失去了缓存的意义。

    看到这里,实际上就是说,二级缓存不能用。整篇文章介绍这么多也没什么用了。


    五、挽救二级缓存?


    想更高效率的使用二级缓存是解决不了了。

    但是解决多表操作避免脏数据还是有法解决的。解决思路就是通过拦截器判断执行的sql涉及到那些表(可以用jsqlparser解析),然后把相关表的缓存自动清空。但是这种方式对缓存的使用效率是很低的。

    设计这样一个插件是相当复杂的,既然我没想着去实现,就不废话了。

    最后还是建议,放弃二级缓存,在业务层使用可控制的缓存代替更好。
  • 相关阅读:
    Debian8搭建LEMP环境
    ProjectManager Beta 7 项目管理器发布
    我的Linux软件集
    修改/home内子目录的名字
    Nginx配置特定二级域名
    Debian8 安装wordpress博客
    LinuxMint18使用单独分区作为Home挂载点
    LinuxMint18配置Grub2默认启动操作系统
    《失恋33天》从绝境中走出来的故事
    爱的世界很拥挤,写在读《爱,就这么简单》之后
  • 原文地址:https://www.cnblogs.com/KingIceMou/p/9389872.html
Copyright © 2011-2022 走看看