美文网首页SSM
Web框架-Mybatis-Mybatis缓存

Web框架-Mybatis-Mybatis缓存

作者: HughJin | 来源:发表于2021-04-05 06:45 被阅读0次

    Java工程师知识树 / Web框架SSM


    缓存技术是一种“以空间换时间”的设计理念,是利用内存空间资源来提高数据检索速度的有效手段之一。Mybatis包含一个非常强大的查询缓存特性,可以非常方便地配置和定制。MyBatis将数据缓存设计成两级结构,分为一级缓存、二级缓存。

    MyBatis一级缓存

    默认情况下,MyBatis只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存。这也就是大家常说的MyBatis一级缓存,一级缓存的作用域是SqlSession

    MyBatis一级缓存的运行过程是这样的:执行SQL语句的过程中,首次执行它时从数据库获取的所有数据会被存储在一段高速缓存中,今后执行这条语句时就会从高速缓存中读取结果,而不是再次查询数据库。 MyBatis提供了默认下基于Java HashMap的缓存实现。

    一级缓存总结:MyBatis执行SQL语句之后,这条语句的执行结果被缓存,以后再执行这条语句的时候,会直接从缓存中拿结果,而不是再次执行SQL。但是一旦执行新增或更新或删除操作,缓存就会被清除。

    Mybatis缓存使用情况 结论
    同个session进行两次相同查询 MyBatis只进行1次数据库查询
    同个session进行两次不同的查询 MyBatis进行两次数据库查询
    不同session,进行相同查询 MyBatis进行两次数据库查询
    同个session,查询之后更新数据,再次查询相同的语句 更新操作之后缓存会被清除

    MyBatis二级缓存

    缓存,这个伟大的创新,无处不在,而且还喜欢分级,往往以多级缓存的形式出现。mybatis的缓存如出一辙,也划分为一级缓存和二级缓存。

    1、CPU的二级缓存

    缓存无处不在,其最根本的起源是CPU的多级缓存,如下所示:

    CPU的二级缓存

    对比一下CPU的二级缓存,更有助于学习MyBatis二级缓存。

    二级缓存是CPU性能表现的关键之一,在CPU核心不变化的情况下,增加二级缓存容量能使性能大幅度提高。而同一核心的CPU高低端之分往往也是在二级缓存上有差异,由此可见二级缓存对于CPU的重要性。

    2、MyBatis二级缓存出现的原因

    之所以称之为“二级缓存”,是相对于“一级缓存”而言的。既然有了一级缓存,那么为什么要提供二级缓存呢?

    我们知道,在一级缓存中,不同session进行相同SQL查询的时候,是查询两次数据库的。显然这是一种浪费,既然SQL查询相同,就没有必要再次查库了,直接利用缓存数据即可,这种思想就是MyBatis二级缓存的初衷

    另外,SpringMyBatis整合时,每次查询之后都要进行关闭sqlsession,关闭之后数据被清空。所以MyBatisSpring整合之后,一级缓存是没有意义的。如果开启二级缓存,关闭sqlsession后,会把该sqlsession一级缓存中的数据添加到mapper namespace的二级缓存中。这样,缓存在sqlsession关闭之后依然存在。

    3、MyBatis二级缓存的设置

    默认情况下,MyBatis只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存,见: MyBatis一级缓存介绍

    要启用全局的二级缓存,

    SqlMapConfig.xml内配置:<setting name="cacheEnabled" value="true"/>

    在SQL映射文件中添加:

    <!-- cacahe节点:配置当前命名空间下mapper的缓存 -->
    <cache></cache> <!-- 使用默认值 -->
    <cache eviction= "LRU" flushInterval= "100000" readOnly= "true" size= "1024"/> <!-- 使用指定值 -->
    

    eviction:其中eviction代表缓存回收策略,目前mybatis提供以下回收策略:

    • LRU(Least Recently Used):最近最少使用的,回收最长时间不用的对象;
    • FIFO(First in first out):先进先出,按照对象进入缓存的顺序来移除;
    • SOFT:软引用,移除基于垃圾回收器状态和软引用规则的对象;
    • WEAK:弱引用,更积极的移除基于垃圾收集器状态和弱引用规则的对象。

    flushInterval:缓存刷新时间间隔,单位为毫秒,每经过相应时间会对缓存进行刷新,如果没有配置,当SQL被执行的时候才会刷新缓存。

    size:引用数目,是一个整数,代表缓存最多可以存储多少的对象,该数据不宜设置过大,如果设置过大会导致内存溢出。

    readOnly:只读,设置只读为“true”意味着缓存数据只能读取而不能修改,这样设置的好处是我们可以快速读取缓存,缺点是没有办法修改缓存,它的默认值是false,不允许修改。

    提示:二级缓存是事务性的。这意味着,当 SqlSession 完成并提交时,或是完成并回滚,但没有执行 flushCache=true 的 insert/delete/update 语句时,缓存会获得更新。

    二级缓存是Mapper级别的缓存,多个SqlSession去操作同一个Mappersql语句,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。

    4、MyBatis 使用自定义缓存

    除了上述自定义缓存的方式,你也可以通过实现自己的缓存,或为其他第三方缓存方案创建适配器,来完全覆盖缓存行为。

    <cache type="cn.mybatis.mydemo.mycache.MyCustomCache"/>
    

    这个示例展示了如何使用一个自定义的缓存实现。type属性指定的类必须实现 org.mybatis.cache.Cache 接口,且提供一个接受 String参数作为 id 的构造器。 这个接口是 MyBatis框架中许多复杂的接口之一,但是行为却非常简单,代码如下所示:

    public interface Cache {
      String getId();
      int getSize();
      void putObject(Object key, Object value);
      Object getObject(Object key);
      boolean hasKey(Object key);
      Object removeObject(Object key);
      void clear();
    }
    

    为了对你的缓存进行配置,只需要简单地在你的缓存实现类中添加公有的 JavaBean属性,然后通过 cache元素传递属性值,例如,下面的例子将在你的缓存实现类上调用一个名为 setCacheFile(String file) 的方法:

    <cache type="cn.mybatis.mydemo.mycache.MyCustomCache">
      <property name="cacheFile" value="/tmp/my-custom-cache.tmp"/>
    </cache>
    

    你可以使用所有简单类型作为JavaBean属性的类型,MyBatis会进行转换。你也可以使用占位符(如 ${cache.file}),以便替换成在配置文件属性中定义的值。

    从版本 3.4.2 开始,MyBatis已经支持在所有属性设置完毕之后,调用一个初始化方法。这样就可以在上面定义的MyCustomCache里面获取配置文件设置的cacheFile属性。如果想要使用这个特性,请在你的自定义缓存类里实现 org.apache.ibatis.builder.InitializingObject 接口。

    public interface InitializingObject {
      void initialize() throws Exception;
    }
    

    提示:上文中对缓存的配置(如清除策略、可读或可读写等),不能应用于自定义缓存。

    请注意,缓存的配置和缓存实例会被绑定到 SQL 映射文件的命名空间中。 因此,同一命名空间中的所有语句和缓存将通过命名空间绑定在一起。 每条语句可以自定义与缓存交互的方式,或将它们完全排除于缓存之外,这可以通过在每条语句上使用两个简单属性来达成。 默认情况下,语句会这样来配置:

    <select ... flushCache="false" useCache="true"/><!--select查询的结果会使用缓存-->
    <insert ... flushCache="true"/><!--insert插入操作会清空缓存-->
    <update ... flushCache="true"/>
    <delete ... flushCache="true"/>
    

    鉴于这是默认行为,显然你永远不应该以这样的方式显式配置一条语句。但如果你想改变默认的行为,只需要设置 flushCacheuseCache属性。比如,某些情况下你可能希望特定 select语句的结果排除于缓存之外,或希望一条 select语句清空缓存。类似地,你可能希望某些 update语句执行时不要刷新缓存。

    4、MyBatis二级缓存的复用:cache-ref

    对某一命名空间的语句,只会使用该命名空间的缓存进行缓存或刷新。但你可能会想要在多个命名空间中共享相同的缓存配置和实例。要实现这种需求,你可以使用 cache-ref 元素来引用另一个缓存。

    <cache-ref namespace="cn.mybatis.mydemo.mapper.XXXMapper"/>
    

    相关文章

      网友评论

        本文标题:Web框架-Mybatis-Mybatis缓存

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