二级缓存原理
首先开启mybatis的二级缓存。
SqlSession1去查询用户id为1的用户信息,查询到用户信息会将查询数据存储到二级缓存中。
如果SqlSession3去执行相同mapper下sql,执行commit提交,清空该mapper下的二级缓存区域的数据。
SqlSession2去查询用户id为1的用户信息,去缓存中找是否存在数据,如果存在则直接从缓存中取出数据。
二级缓存与一级缓存区别:二级缓存的区域更大,多个SqlSession可以共享一个UserMapper的二级缓存区域。UserMapper有一个二级缓存区域(按namespace区分),其他mapper也有自己的二级缓存区域。
每一个namespace的mapper都有一个二级缓存区域,两个mapper的namespace如果相同,这两个mapper执行Sql查询到数据将存到同一个二级缓存区域中。
开启二级缓存
mybatis的二级缓存是mapper范围级别,在SqlMapConfig.xml开启二级缓存
在UserMapper.xml中开启二级缓存,UserMapper.xml下的sql执行完成会存储到它的缓存区域(HashMap) 。
调用pojo类实现序列化接口
为了将缓存数据取出执行序列化操作, 因为二级缓存数据存储介质多种多样,不一定在内存中。
测试方法
//二级缓存测试
@Test
public void testCache2() throws Exception {
SqlSession sqlSession1 = sqlSessionFactory.openSession();
SqlSession sqlSession2 = sqlSessionFactory.openSession();
SqlSession sqlSession3 = sqlSessionFactory.openSession();
UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class);
UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class);
UserMapper userMapper3 = sqlSession3.getMapper(UserMapper.class);
//第一次发起请求,查询id为1的用户
User user1 = userMapper1.findUserById(1);
System.out.println(user1);
//这里执行关闭操作,将SqlSession中的数据写到二级缓存区域
sqlSession1.close();
//使用SqlSession执行commit操作
User user = userMapper3.findUserById(1);
user.setUsername("CHLOE");
userMapper3.updateUser(user);
sqlSession3.commit();
sqlSession3.close();
//第二次发起请求,查询id为1的用户
User user2 = userMapper2.findUserById(1);
System.out.println(user2);
sqlSession2.close();
}
useCache配置
在statement中设置useCache=false可以禁用当前的select语句的二级缓存,即每次查询都会发出sql去查询,默认情况是true,即该sql使用二级缓存。
<select id="findOrderListResultMap" resultMap="OrderUserMap" useCache="false">
总结:针对每次查询都需要最新的数据sql,要设置成useCache=false,禁用二级缓存。
刷新缓存(就是清空缓存)
在mapper的同一个namespace中,如果有其他insert、update、delete操作数据后需要刷新缓存,如果不执行刷新缓存会出现脏读。
在statement中设置flushCache="true"属性,默认情况下为true即刷新缓存,如果改成false则不会刷新。使用缓存时如果手动修改数据表中的查询数据会查询脏读。
<select id="findOrderListResultMap" resultMap="OrderUserMap" flushCache="true">
总结:一般执行完commit操作都需要刷新缓存,flushCache=true表示刷新缓存,这样可以避免数据库脏读。