MyBatis 缓存机制

MyBatis 缓存机制

简介

MyBatis 包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制。缓存可以极大的提升查询效率.
MyBatis系统中默认定义了两级缓存:一级缓存、二级缓存

  1. 默认情况下,只有一级缓存(SqlSession级别的缓存,也称为本地缓存)开启。
  2. 二级缓存需要手动开启和配置,他是基于namespace级别的缓存。

一级缓存

  1. 一级缓存(local cache), 即本地缓存, 作用域默认为sqlSession。同一次会话期间只要查询过的数据都会保存在当前SqlSession的一个Map中。

  2. 级缓存失效的几种情况
    不同的SqlSession对应不同的一级缓存
    同一个SqlSession但是查询条件不同
    同一个SqlSession两次查询期间执行了任何一次增删改操作
    同一个SqlSession两次查询期间手动清空了缓存

@Test

     public void testFirstCache() throws Exception {


          SqlSessionFactory sqlSessionFactory = 
getSqlSessionFactory();

          SqlSession sqlSession1 = 
sqlSessionFactory.openSession(true);

          EmpMapper mapper1 = 
sqlSession1.getMapper(EmpMapper.class);

          Emp emp1 = mapper1.getEmpByEid("13");

          System.out.println(emp1);

          

          System.out.println("********************");

          mapper1.deleteMoreEmp("1");

          sqlSession1.clearCache();

          

          SqlSession <u>sqlSession2</u> = 
sqlSessionFactory.openSession(true);

          EmpMapper mapper2 = 
sqlSession1.getMapper(EmpMapper.class);

          Emp emp2 = mapper2.getEmpByEid("13");

          System.out.println(emp2);

     }


二级缓存

  • 二级缓存(second level cache),全局作用域缓存

  • 二级缓存默认不开启,需要手动配置

  • MyBatis提供二级缓存的接口以及实现,缓存实现要求POJO实现Serializable接口

  • 二级缓存在 SqlSession 关闭或提交之后才会生效

  • 二级缓存使用的步骤:
    全局配置文件中开启二级缓存

<!-- 开启二级缓存 -->

          <setting name="cacheEnabled" value="true"/>


需要使用二级缓存的映射文件处使用cache配置缓存

<!-- 开启二级缓存 -->

     <cache 
type="<u>org.mybatis.caches.ehcache.EhcacheCache</u>"/>


注意:POJO需要实现Serializable接口

public class Emp implements Serializable{

}

缓存相关的属性

  1. 全局setting的cacheEnable:
    配置二级缓存的开关,一级缓存一直是打开的
  2. select标签的useCache属性:
    配置这个select是否使用二级缓存。一级缓存一直是使用的
  3. sql标签的flushCache属性:
    增删改默认flushCache=true。sql执行以后,会同时清空一级和二级缓存。查询默认 flushCache=false。
  4. sqlSession.clearCache():只是用来清除一级缓存
  5. readOnly:只读,true/false
    true:只读缓存;会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。
    false:读写缓存;会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是 false。
@Test

     public void testSecondCache() throws Exception {

          SqlSessionFactory sqlSessionFactory = 
getSqlSessionFactory();

          SqlSession sqlSession = 
sqlSessionFactory.openSession(true);

          EmpMapper mapper1 = 
sqlSession.getMapper(EmpMapper.class);

          Emp emp1 = mapper1.getEmpByEid("13");

          System.out.println(emp1);

          sqlSession.commit();

          //删除功能:清除缓存

          /* mapper1.deleteMoreEmp("1"); */

          //只是影响一级缓存

          sqlSession.clearCache();

          System.out.println("===================");

          

          EmpMapper mapper2 = 
sqlSession.getMapper(EmpMapper.class);

          Emp emp2 = mapper2.getEmpByEid("13");

          System.out.println(emp2);

          

     }


posted @ 2020-07-18 10:36  流沙uiui  阅读(105)  评论(0编辑  收藏  举报