-
一级缓存:线程级别的缓存,本地缓存;SqlSession级别的缓存;
-
二级缓存:全局范围的缓存;除当前线程sqlsession能用外,其他也可以使用的缓存。
mybatis的缓存机制可以理解为一个map集合,能够保存并查询出一些数据;
一级缓存:mybatis默认情况下就是一级缓存;对于查询过的数据,mybatis会保存在一个缓存中(Map);当下次再获取同样的数据时,可以直接从一级缓存中拿。
@Test
public void test(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
TeacherMapper mapper = sqlSession.getMapper(TeacherMapper.class);
Teacher teacher1 = mapper.getTeacher(1);
System.out.println(teacher1);
System.out.println("--------------------------");
Teacher teacher2 = mapper.getTeacher(1);
System.out.println(teacher2);
System.out.println(teacher1==teacher2);//true
sqlSession.close();
}
但在以下几种情况下一级缓存会失效。
1、 使用不同的sqlsession。(使用多个Mapper,查询同一个对象,这样一级缓存会失效)只有在同一个sqlsession期间查询到的数据会保存到这个SqlSession的缓存中;下次使用这个sqlsession查询会从缓存中拿。
2、 执行同一个SQL语句,当传入的参数与之前不同,那么缓存就不会生效。
3、 在这个sqlsession期间执行任何一个增删改操作,就会清空以前的缓存。这样做的目的是为了让缓存中存储最新的信息。避免脏读!
4、 手动清空当前sqlsession的缓存。
sqlSession.clearCache();//手动清除缓存
二级缓存:全局作用域缓存,他默认不开启,需要手动设置;mybatis提供二级缓存的接口以及实现,缓存时限要求pojo实现Serializable接口。
二级缓存在sqlsession关闭或提交之后才会生效。
使用步骤
- 全局配置中开启二级缓存
- 需要使用二级缓存的映射文件处使用cache配置缓存
- 注意:pojo需要实现Serializable接口。(持久化)
<!--开启二级缓存-->
<setting name="cacheEnabled" value="true"/>
在Mapper.xml中
<!--在本Mapper中开启默认二级缓存-->
<cache></cache>
记得要让pojo持久化,实现Serializable接口。
@Test
public void test(){
SqlSession sqlSession1 = MybatisUtils.getSqlSession();
SqlSession sqlSession2 = MybatisUtils.getSqlSession();
TeacherMapper mapper = sqlSession1.getMapper(TeacherMapper.class);
TeacherMapper mapper2 = sqlSession2.getMapper(TeacherMapper.class);
Teacher teacher1 = mapper.getTeacher(1);
System.out.println(teacher1);
sqlSession1.close();
System.out.println("--------------------------");
Teacher teacher2 = mapper2.getTeacher(1);
System.out.println(teacher2);
sqlSession2.close();
}
输出结果:
Teacher{tid=0, name='null', students=[Student{id=1, name='阿涛', tid=1}, Student{id=2, name='洋屁', tid=1}]}
Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@6950e31]
Returned connection 110431793 to pool.
--------------------------
Cache Hit Ratio [cn.edu.nyist.mybatis.dao.TeacherMapper]: 0.3333333333333333//命中率
Teacher{tid=0, name='null', students=[Student{id=1, name='阿涛', tid=1}, Student{id=2, name='洋屁', tid=1}]}
二级缓存是namespace级别的缓存。
-
不会出现一级缓存和二级缓存都有同一个数据。
-
二级缓存中:一级缓存关闭了就有了;存储到二级缓存
-
一级缓存中:二级缓存如果没有数据,就会看一级缓存中有没有,如果一级缓存还没有就会向数据库中查询,查找到后放到一级缓存中。
-
任何时候都是先查看二级缓存,再查看一级缓存,最后在从数据库中查。
二级缓存机制的应用场景:
对于访问多的查询请求且用户对查询结果实时性要求不高,此时可采用mybatis二级缓存技术降低数据库访问量,提高访问速度,业务场景比如:耗时较高的统计分析sql、电话账单查询sql等。实现方法如下:通过设置刷新间隔时间,由mybatis每隔一段时间自动清空缓存,根据数据变化频率设置缓存刷新间隔flushInterval,比如设置为30分钟、60分钟、24小时等,根据需求而定。
缓存原理示意图:
整合第三方缓存:
Mybatis:自带的缓存–Cache;允许实现cache接口,自定义第三方缓存。
对数据进行集中管理的缓存框架:Redis、memcached、ehcache…
这里主要介绍ehcache缓存框架。
整合ehcache;ehcache是一个专业级的Java进程内的缓存框架;
1.
导包。
ehcache核心包:ehcache-core-2.6.8.jar
mybatis-ehcache-1.0.3.jar(ehcache的整合包)
slf4j-api-1.7.21.jar
slf4j-log4j12-1.7.12.jat;
- 配置文件
ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
<diskStore path="F:developehcache"/>
<defaultCache
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
maxElementsOnDisk="10000000"
diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU">
<persistence strategy="localTempSwap"/>
</defaultCache>
</ehcache>
配置mapper中cache中的type为ehcache对cache接口的实现类型。ehcache对cache接口有一个实现类为
useCache默认是true,用来设置是否禁用二级缓存的,在statement中设置useCache=false可以禁用当前select语句的二级缓存,每次查询都会发出SQL查询,默认情况下是true,使用二级缓存。
<select id="getTeacher" resultMap="teacherAndStudent" useCache="false">
select * from teacher where tid = #{tid}
</select>
如果设置成上面的格式,也就是说每次从teacher表中查数据都会访问数据库,而不是用二级缓存。insert、delete、update操作数据都需要刷新缓存,如果不刷新缓存,就会产生脏读。flushCache=“true” 属性默认是true,刷新缓存,如果改为false就不会刷新。