mybatis缓存机制

  • 一级缓存:线程级别的缓存,本地缓存;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级别的缓存。

  1. 不会出现一级缓存和二级缓存都有同一个数据。

  2. 二级缓存中:一级缓存关闭了就有了;存储到二级缓存

  3. 一级缓存中:二级缓存如果没有数据,就会看一级缓存中有没有,如果一级缓存还没有就会向数据库中查询,查找到后放到一级缓存中。

  4. 任何时候都是先查看二级缓存,再查看一级缓存,最后在从数据库中查。

二级缓存机制的应用场景:
对于访问多的查询请求且用户对查询结果实时性要求不高,此时可采用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;
  1. 配置文件
    ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
    <diskStore path="F:\develop\ehcache"/>
    <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就不会刷新。

posted on 2019-12-04 15:42  青山是谁  阅读(16)  评论(0编辑  收藏  举报