MyBatis缓存介绍

一、MyBatis缓存介绍
正如大多数持久层框架一样,MyBatis同样提供了一级缓存和二级缓存的支持
1.一级缓存:基于PerpetualCache的HashMap本地缓存,其存储作用域为SqlSession,当Session flush或close后,该Session中的所有Cache就将清空;
2.二级缓存:与一级缓存机制相同,默认也是采用PerpetualCache,HashMap存储,不同在于其存储作用域为Mapper(Namespace),并可自定义存储源,如Ehcache;
3.对于缓存数据的更新机制,当某个作用域(一级缓存Session/二级缓存Namespace)进行了C/U/D操作后,默认该作用域下所有缓存将被clear。

二、MyBatis一级缓存测试
public class TestOneLevelCache {

    /**
     * 一级缓存,也就是Session级的缓存,默认开启
     */
    @Test
    public void testCache1() {
        SqlSession sqlSession = MyBatisUtil.getSqlSession(true);
        String statement = "com.mapping.teacherMapper.selectTeacher";
        Teacher t = sqlSession.selectOne(statement, 3);
        System.out.println(t);
        
        /*
         * 一级缓存默认就会被使用
         */
        t = sqlSession.selectOne(statement, 3);
        sqlSession.close();
        System.out.println(t);
        /*
         * 1.必须是同一个Session,如果Session对象已经close过了就不能再被使用了
         */
        sqlSession = MyBatisUtil.getSqlSession(true);
        t = sqlSession.selectOne(statement, 3);
        System.out.println(t);
        /*
         * 2.查询条件不同
         */
        t = sqlSession.selectOne(statement, 4);
        System.out.println(t);
        /*
         * 3.清理缓存
         */
        sqlSession.clearCache();
        t = sqlSession.selectOne(statement, 4);
        System.out.println(t);
        /*
         * 4.增删改的操作(这些操作都会清理缓存)
         */
        Teacher tt = new Teacher(3, "hejiong");
        sqlSession.update("com.mapping.teacherMapper.updateTeacher", tt);
        t = sqlSession.selectOne(statement, 3);
        System.out.println(t);
        sqlSession.close();
    }
}
对应的updateTeacher配置:
<update id="updateTeacher" parameterType="com.domain.Teacher">
        update ksl_teacher
        <set>
            tname=#{name}
        </set>
        where tid=#{id}
</update>
注:Teacher类在添加了(int, String)构造函数后,就没有了默认的无参构造函数,此时需要将无参构造函数添加上,否则会报错。

三、MyBatis二级缓存测试
1.开启二级缓存
<!-- 开启二级缓存 -->
<cache/>
2.测试二级缓存
public class TestTwoLevelCache {

    /**
     * 测试二级缓存
     * 使用两个不同的SqlSession对象去执行相同查询条件的查询,第二次查询时不会再发送SQL语句,而是直接从缓存中取出数据
     */
    @Test
    public void testCache2() {
        SqlSessionFactory factory = MyBatisUtil.getSqlSessionFactory();
        String statement = "com.mapping.teacherMapper.selectTeacher";
        // 开启两个不同的SqlSession
        SqlSession session1 = factory.openSession();
        SqlSession session2 = factory.openSession();
        // 使用二级缓存时,实体类必须实现Serializable接口
        Teacher t = session1.selectOne(statement, 1);
        session1.commit();// 此处必须提交事务后,二级缓存才起作用
        System.out.println(t);
        
        // 由于使用的不是同一个SqlSession对象,所以即使查询条件相同,一级缓存也不会开启使用
        t = session2.selectOne(statement, 1);
        System.out.println(t);
    }
}
注:使用二级缓存时,实体类必须实现Serializable接口
3.二级缓存补充说明
映射语句文件中的所有select语句将会被缓存;
映射语句文件中的所有insert/update和delete语句会刷新缓存;
缓存会使用Least Recently Used(LRU,最近最少使用的)算法来收回;
缓存会根据指定的时间间隔来刷新;
缓存会存储1024个对象。

cache标签常用属性:
<cache
eviction="FIFO" <!--回收策略为先进先出-->
flushInterval="60000" <!--自动刷新时间为60s-->
size="512" <!--最多缓存512个引用对象-->
readOnly="true"/> <!--只读-->

上一篇:java – 无法运行JDI跟踪示例:错误:无法找到或加载主类


下一篇:mac os mysql 5.7.25 安装教程