MyBatis 缓存机制
简介
MyBatis 包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制。缓存可以极大的提升查询效率.
MyBatis系统中默认定义了两级缓存:一级缓存、二级缓存
- 默认情况下,只有一级缓存(SqlSession级别的缓存,也称为本地缓存)开启。
- 二级缓存需要手动开启和配置,他是基于namespace级别的缓存。
一级缓存
-
一级缓存(local cache), 即本地缓存, 作用域默认为sqlSession。同一次会话期间只要查询过的数据都会保存在当前SqlSession的一个Map中。
-
级缓存失效的几种情况
不同的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{
}
缓存相关的属性
- 全局setting的cacheEnable:
配置二级缓存的开关,一级缓存一直是打开的 - select标签的useCache属性:
配置这个select是否使用二级缓存。一级缓存一直是使用的 - sql标签的flushCache属性:
增删改默认flushCache=true。sql执行以后,会同时清空一级和二级缓存。查询默认 flushCache=false。 - sqlSession.clearCache():只是用来清除一级缓存
- 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);
}