跳到主要内容

15、MyBatis 实战 - 之MyBatis二级缓存

MyBatis二级缓存

1、二级缓存的范围

Mybatis的一级缓存的范围是SqlSession,而二级缓存的范围是SqlSessionFactory,二级缓存的范围更大,相当于是数据库级别,一级缓存相当于表级别。

2、使用二级缓存需要具备的条件

(1)<setting name="cacheEnabled" value="true">
全局性地开启或关闭所有映射器配置文件中已配置的任何缓存。默认就是true,
无需设置,

(2)在需要使用二级缓存的SqlMapperxml文件中添加配置:<cache/>
(3)使用二级缓存的实体类对象必须是可序列化的,
也就是必须实现iava.io.Serializable接口
(4)SqlSession对象关闭或提交之后,一级缓存中的数据才会被写入到二级缓存当中。
此时二级缓存才可用。

3、二级缓存代码演示

pojo类Clazz ,记住要实现序列化接口Serializable

public class Clazz implements Serializable {
   
     
    private Integer cid;
    private String name;
    ......此处省略get、set方法

ClazzMapper 接口

public interface ClazzMapper {
   
     

    public Clazz selectByCidStep2(Integer cid);
}

ClazzMapper.xml ,记得加上 cache 标签

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.powernode.mybatis.mapper.ClazzMapper">

    <cache/>
    <select id="selectByCidStep2" resultType="Clazz">
        select * from t_clazz where cid ={cid}
    </select>
</mapper>

我们通过测试类来趴一下

 @Test
    public void testSecondCache() throws Exception {
   
     
        SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));
        SqlSession sqlSession1 = factory.openSession();
        ClazzMapper mapper = sqlSession1.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        sqlSession1.close();
        SqlSession sqlSession2 = factory.openSession();
        ClazzMapper mapper1 = sqlSession2.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession2.close();
    }

运行结果如下,第一次查询,没有缓存,所以我们看到:
Cache Hit Ratio [com.powernode.mybatis.mapper.ClazzMapper]: 0.0
表示缓存命中率为0.0,第二次查询,因为开启了二级缓存,所以命中率变成了Cache Hit Ratio [com.powernode.mybatis.mapper.ClazzMapper]: 0.5
,此时直接走了缓存取数据,并没有查sql语句
 
我们发现上面的测试类,使用的是两个不同的SqlSession 执行对象,如果没有开启二级缓存,那么肯定每一次都会去查询sql语句,但是现在加了二级缓存的配置,所以就走了缓存。

需要注意的一点是:如果SqlSession 对象没有关闭,那么就不会走二级缓存,

我们可以通过代码来演示一下,代码有一点变化,就是去掉了 sqlSession1.close();

@Test
    public void testSecondCache() throws Exception {
   
     
        SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));
        SqlSession sqlSession1 = factory.openSession();
        ClazzMapper mapper = sqlSession1.getMapper(ClazzMapper.class);
        Clazz clazz = mapper.selectByCidStep2(1000);
        System.out.println(clazz);
        SqlSession sqlSession2 = factory.openSession();
        ClazzMapper mapper1 = sqlSession2.getMapper(ClazzMapper.class);
        Clazz clazz1 = mapper1.selectByCidStep2(1000);
        System.out.println(clazz1);
        sqlSession2.close();
    }

运行结果如下,我们发现两次的查询缓存命中率都为0.0,都没有走缓存,原因就是因为没有关闭SqlSession 对象。所以要开启二级缓存,除了必要的文件配置,还需要在使用的时候,一定要关闭SqlSession 对象,这样二级缓存才会生效
 

4、二级缓存的失效

二级缓存的失效:只要两次查询之间出现了增删改操作。二级缓存就会失效。【一级缓存也会失效】