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、二级缓存的失效
二级缓存的失效:只要两次查询之间出现了增删改操作。二级缓存就会失效。【一级缓存也会失效】