zoukankan      html  css  js  c++  java
  • 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: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就不会刷新。

  • 相关阅读:
    如何使用Shiro
    ORACLE: 查询(看)表的主键、外键、唯一性约束和索引
    图片下载器类
    关于Android如何创建空文件夹,以及mkdir和mkdirs的区别
    图片二值化 和灰度处理方法
    InputSream转为String
    Bitmap Byte[] 互转
    静默安装/ 普通安装与root权限获取相关
    EventBus 3.0使用相关
    文件存储工具类
  • 原文地址:https://www.cnblogs.com/dataoblogs/p/14121983.html
Copyright © 2011-2022 走看看