MyBatis 的缓存机制

143 阅读3分钟

MyBatis 的缓存机制

缓存:cache

缓存的作用:通过减少IO的方式,来提高程序的执行效率。

mybatis的缓存:将 select 语句的查询结果放到缓存(内存)当中,下一次还是这条 select 语句的话,直接从缓存中取,不再查数据库。一方面是减少了IO。另一方面不再执行繁琐的查找算法。效率大大提升。

mybatis缓存包括:

  • 一级缓存:将查询到的数据存储到 SqlSession 中。
  • 二级缓存:将查询到的数据存储到 SqlSessionFactory 中。
  • 或者集成其它第三方的缓存:比如EhCache【Java语言开发的】、Memcache【C语言开发的】等。

缓存只针对于DQL语句,也就是说缓存机制只对应select语句。

1. 一级缓存

  1. 一级缓存在 MyBatis 中是默认开启的,不需要任何配置。
  2. 在使用同一个 SqlSession 查询相同的数据时,通过观察日志,会发现只会查询一次。
  3. 什么情况下不会走缓存:
  • 不同的 SqlSession 对象
  • 查询条件发生改变
  1. 缓存失效的原因:
  • 两次相同查询期间,使用 sqlSession.clearCache() 方法手动清空了一级缓存

  • 两次查询期期间,进行了增删改操作。这个增删改和哪张表没有关系,只要有 insert delete update 操作,一级缓存就失效。

2. 二级缓存

  • 二级缓存的范围是 SqlSessionFactory。

  • 使用二级缓存需要具备以下几个条件:

    1. 全局性地开启或关闭所有映射器配置文件中已配置的任何缓存。默认就是true,无需设置。
    2. 在需要使用二级缓存的SqlMapper.xml文件中添加配置:
    3. 使用二级缓存的实体类对象必须是可序列化的,也就是必须实现java.io.Serializable接口
    4. SqlSession对象关闭或提交之后,一级缓存中的数据才会被写入到二级缓存当中。此时二级缓存才可用。
  • 二级缓存失效的原因:两次查询期间,进行了增删改操作,一级和二级缓存都会失效。

  • 在使用 标签开启二级缓存时,可以配置其中的参数。

    1. eviction:指定从缓存中移除某个对象的淘汰算法。默认采用LRU策略。

      1. LRU:Least Recently Used。最近最少使用。优先淘汰在间隔时间内使用频率最低的对象。(其实还有一种淘汰算法LFU,最不常用。)
      2. FIFO:First In First Out。一种先进先出的数据缓存器。先进入二级缓存的对象最先被淘汰。
      3. SOFT:软引用。淘汰软引用指向的对象。具体算法和JVM的垃圾回收算法有关。
      4. WEAK:弱引用。淘汰弱引用指向的对象。具体算法和JVM的垃圾回收算法有关。
    2. flushInterval:

      1. 二级缓存的刷新时间间隔。单位毫秒。如果没有设置。就代表不刷新缓存,只要内存足够大,一直会向二级缓存中缓存数据。除非执行了增删改。
    3. readOnly:

      1. true:多条相同的sql语句执行之后返回的对象是共享的同一个。性能好。但是多线程并发可能会存在安全问题。
      2. false:多条相同的sql语句执行之后返回的对象是副本,调用了clone方法。性能一般。但安全。
    4. size:

      1. 设置二级缓存中最多可存储的java对象数量。默认值1024。