2019-03-23 查询缓存-二级缓存

二级缓存原理

首先开启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表示刷新缓存,这样可以避免数据库脏读。

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容