Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存

摘要:
请注明转载来源:http://www.cnblogs.com/Joanna-Yan/p/6956206.html如前所述:Spring+SpringMVC+MyBatis深度学习和构建(7)-MyBatis延迟加载1.MyBatis提供的查询缓存是什么,用于减轻数据库压力和提高数据库性能。Mybatis提供一级缓存和二级缓存。第一级缓存是SqlSession级缓存。操作数据库时需要构造

转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/6956206.html 

前面讲到:Spring+SpringMVC+MyBatis深入学习及搭建(七)——MyBatis延迟加载

1.什么是查询缓存

mybatis提供查询缓存,用于减轻数据库压力,提高数据库性能。

mybatis提供一级缓存和二级缓存。

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第1张

一级缓存是SqlSession级别的缓存。在操作数据库时需要构造sqlSession对象,在对象中有一个数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的。

二级缓存是mapper级别的缓存,多个sqlSession去操作同一个Mapper的sql语句,多个sqlSession可以共用二级缓存,二级缓存是跨sqlSession的。

为什么要用缓存?

如果缓存中有数据就不用从数据库中获取,大大提高系统性能。

2.一级缓存

2.1一级缓存工作原理

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第2张

 第一次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,如果没有,从数据库查询用户信息。

得到用户信息,将用户信息存储到一级缓存中。

如果sqlSession去执行commit操作(执行插入、更新、删除),清空sqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。

第二次发去查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,缓存中有,直接从缓存中获取用户信息。

 2.2一级缓存测试

 mybatis默认支持一级缓存,不需要在配置文件去配置。

 按照上边一级缓存原理步骤去测试。

    @Test
    public void testCache1() throws Exception{
        SqlSession sqlSession=sqlSessionFactory.openSession();
        UserMapper userMapper=sqlSession.getMapper(UserMapper.class);
        //下边查询使用一个SqlSession
        //第一次发起请求,查询id为1的用户
        User user1=userMapper.findUserById(1);
        System.out.println(user1);
        
        //如果sqlSession去执行commit操作(执行插入、更新、删除),清空sqlSession中的一级缓存,
        //这样做的目的是为了让缓存中存储的是最新的信息,避免脏读。
        //更新user1的信息
        user1.setUsername("测试用户22");
        userMapper.updateUser(user1);
        //执行commit操作去清空缓存
        sqlSession.commit();
        
        //第二次发起请求,查询id为1的用户
        User user2=userMapper.findUserById(1);
        System.out.println(user2);

        sqlSession.close();
    }

2.3一级缓存应用

 正式开发,是将mybatis和spring进行整合开发,事务控制在service中。

一个service方法中包括很多Mapper方法调用。

service{

  //开始执行时,开启事务,创建SqlSession对象

  //第一次调用mapper的方法findUserById(1)

  //第二次调用mapper的方法findUserById(1),从一级缓存中取数据

  //方法结束,sqlSession关闭

}

如果是执行两次service调用查询相同的用户信息,不走一级缓存,因为session方法结束,sqlSession就关闭,一级缓存就清空。

3.二级缓存

3.1二级缓存原理

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第3张

首先开启mybatis的二级缓存。

sqlSession1去查询用户id为1的用户信息,查询到用户信息后悔讲查询数据存储到二级缓存中。

如果sqlSession3去执行相同mapper下的sql,执行commit提交,会清空该mapper下的二级缓存区域的数据。

sqlSession2去查询用户id为1的用户信息,去缓存中找是否存在数据,如果存在直接从缓存中取出数据。

二级缓存与一级缓存区别:二级缓存的范围更大,多个sqlSession可以共享一个UserMapper的二级缓存区域。UserMapper有一个二级缓存区域(按namespace分),其它mapper也有自己的二级缓存区域(按namespace分)。

每个namespace的mapper都有一个二级缓存区域,两个mapper的namespace如果相同,这两个mapper执行sql查询到的数据将存在相同的二级缓存区域中。

3.2开启二级缓存

mybatis的二级缓存是mapper范围级别,除了在SqlMapConfig.xml设置二级缓存的总开关,还要在具体的mapper.xml中开启二级缓存。

在核心配置文件SqlMapConfig.xml中加入:

<setting name="cacheEnabled" value="true"/>
 

描述

允许值

默认值

cacheEnabled

对在此配置文件下的所有cache 进行全局性开/关设置。

true false

true

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第4张

在UserMapper.xml中开启二级缓存,UserMapper.xml下的sql执行完成后存储在它的缓存区域(HashMap)。

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第5张

3.3调用pojo类实现序列化接口

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第6张

为了将缓存数据取出执行反序列划操作,因为二级缓存数据存储介质多种多样,不一定在内存。可能在硬盘、远程等。

3.4测试方法

    @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();
        
        //使用sqlSession3执行commit()操作
        User user=userMapper3.findUserById(1);
        user.setUsername("Joanna");
        userMapper3.updateUser(user);
        //执行提交,清空UserMapper下边的二级缓存
        sqlSession3.commit();
        sqlSession3.close();
        
        //第二次发起请求,查询id为1的用户
        User user2=userMapper2.findUserById(1);
        System.out.println(user2);
        sqlSession2.close();
    }

3.5禁用二级缓存

在statement中设置useCache=false可以禁用当前select语句的二级缓存,即每次查询都会发出sql,默认情况是true,即该sql使用二级缓存。

<select resultMap="ordersUserMap" useCache="false">

3.6刷新缓存(就是清空缓存)

 在mapper的同一个namespace中,如果有其它insert、update、delete操作数据后需要刷新缓存,如果不执行刷新缓存会出现脏读。

设置statement配置中的flushCache="true"属性,默认情况下为true即刷新缓存,如果改成false则不会刷新。使用缓存时如果手动修改数据库表中的查询数据会出现脏读。

<insert parameterType="cn.itcast.mybatis.po.User" flushCache="true">

 总结:一般情况下执行完commit操作都需要刷新缓存,flushCache=true表示刷新缓存,这样可以避免数据库脏读。

3.7 Mybatis Cache参数

flushInterval(刷新间隔)可以被设置为任意的正整数,而且它们代表一个合理的毫秒形式的时间端。默认情况是不设置,也局势没有刷新间隔,缓存仅仅调用语句时刷新。

size(引用数目)可以被设置为任意正整数,要记住你缓存的对象数目和你运行环境的可用内存资源数目。默认值是1024。

readOnly(只读)属性可以被设置为true或false。只读的缓存会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。可读写的缓存会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是false。

如下例子:

<cache  eviction="FIFO"  flushInterval="60000"  size="512"  readOnly="true"/>

这个更高级的配置创建了一个 FIFO 缓存,并每隔 60 秒刷新,存数结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此在不同线程中的调用者之间修改它们会导致冲突。可用的收回策略有, 默认的是 LRU:

  1. LRU – 最近最少使用的:移除最长时间不被使用的对象。
  2. FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
  3. SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
  4. WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
4.mybatis整合ehcache

ehcache是一个纯Java的进程内缓存框架,是一种广泛使用的开源Java分布式缓存,具有快速、精干等特点,是Hibernate中默认的CacheProvider。

4.1分布式缓存

为了提高系统并发、性能,一般会系统进行分布式部署(集群部署方式)

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第7张

不使用分布式缓存,缓存的数据在各个服务单独存储,不方便系统开发。所以要使用分布式缓存对缓存数据进行集中管理。

mybatis的特长是sql操作,缓存数据的管理不是mybatis的特长。mybatis无法实现分布式缓存,需要和其它分布式缓存框架进行整合,如:redis、memcached、ehcache等。

4.2整合方法(掌握)

mybatis提供了一个cache接口,如果要实现自己的缓存逻辑,实现cache接口开发即可。

mybatis和ehcache整合,mybatis和ehcache整合包中提供了一个cache接口的实现类。

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第8张

mybatis默认的cache实现类是:

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第9张

4.3加入ehcache包

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第10张

4.4整合ehcache

配置mapper中cache中的type为ehcache对cache接口的实现类型。

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第11张

Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第12张

4.5加入ehcache的配置文件

在classpath下配置ehcache.xml

<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
    <diskStore path="F:developehcache" />
    <defaultCache 
        maxElementsInMemory="1000" 
        maxElementsOnDisk="10000000"
        eternal="false" 
        overflowToDisk="false" 
        timeToIdleSeconds="120"
        timeToLiveSeconds="120" 
        diskExpiryThreadIntervalSeconds="120"
        memoryStoreEvictionPolicy="LRU">
    </defaultCache>
</ehcache>

属性说明:

diskStore:指定数据在磁盘中的存储位置。

defaultCache:当借助CacheManager.add(“demoCache”)创建Cache时,EhCache便会采用<defaultCache/>指定的管理策略。

以下属性是必须的:

maxElementsInMemory :在内存中缓存的element的最大数目。

maxElementsOnDisk :在磁盘上缓存的element的最大数目,若是0表示无穷大。

eternal :设定缓存的elements是否永远不过期。如果为true,则缓存的数据始终有效,如果为false,你们还要根据timeToIdleSeconds、timeToLiveSeconds判读。

overflowToDisk :设定当内存缓存溢出的时候是否将过期的element缓存到磁盘上。

以下属性是可选的:

timeToIdleSeconds :当缓存在EhCache中的数据前后两次访问的时间超过timeToIdleSeconds的属性取值时,这些数据便会删除,默认值是0,也就是可闲置时间无穷大。

timeToLiveSeconds :缓存element的有效生命期,默认是0,也就是element存活时间无穷大。

diskSpoolBufferSizeMB :这个参数设置DiskStore(磁盘缓存)的缓存区大小。默认是30MB,每个Cache都应该有自己的一个缓冲区。

diskPersistent :在VM重启的时候是否启用磁盘保存EhCache中的数据,默认是false。

diskExpiryThreadIntervalSeconds - 磁盘缓存的清理线程运行间隔,默认是120秒。每个120s,相应的线程会进行一次EhCache中数据的清理工作。

memoryStoreEvictionPolicy - 当内存缓存达到最大,有新的element加入的时候, 移除缓存中element的策略。默认是LRU(最近最少使用),可选的有LFU(最不常使用)和FIFO(先进先出)。

4.6测试程序

同3.4

4.7二级缓存应用场景

 对访问多的查询请求且用户对查询结果实时性要求不高,此时可采用mybatis二级缓存技术降低数据库访问量,提高访问速度,业务场景比如:耗时较高的统计分析sql、电话账单查询sql等。

实现方法如下:通过设置刷新间隔时间,由mybatis每隔一段时间自动清空缓存,根据数据变化频率设置缓存刷新间隔flushInterval,比如设置为30分钟、60分钟、24小时等,根据需求而定。

4.8二级缓存局限性

 mybatis二级缓存对细粒度的数据级别的缓存实现不好,比如如下需求:对商品信息进行缓存,由于商品信息查询访问量大,但是要求用户每次都能查询最新的商品信息,此时如果使用mybatis的二级缓存就无法实现当一个商品变化时只刷新该商品的缓存信息而不刷新其它商品的信息,因为mybatis的二级缓存区域以mapper为单位划分,当一个商品信息变化会将所有商品信息的缓存数据全部清空。解决此类问题需要再业务层根据需求对数据有针对性缓存。

如果此文对您有帮助,微信打赏我一下吧~ 

 Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存第13张

免责声明:文章转载自《Spring+SpringMVC+MyBatis深入学习及搭建(八)——MyBatis查询缓存》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇ASP.NET Core MVC中的IActionFilter.OnActionExecuting方法,可以获取Controller的Action方法参数值List.Select按字符串选择属性下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

SpringBoot+Mybatis-Plus两种分页方法

用到的依赖: <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>${mybatisplus.version}&l...

Spring Boot -- Spring Boot之@Async异步调用、Mybatis、事务管理等

这一节将在上一节的基础上,继续深入学习Spring Boot相关知识,其中主要包括@Async异步调用,@Value自定义参数、Mybatis、事务管理等。 本节所使用的代码是在上一节项目代码中,继续追加的,因此需要先学习上一节内容。 一、使用@Async实现异步调用 要在springboot中使用异步调用方法,只要在被调用的方法上面加上@Async就可以...

SpringBoot初学(4)– JdbcTemplate和Mybatis

前言 github: https://github.com/vergilyn/SpringBootDemo 代码位置: 一、Spring Boot集成JdbcTemplate或NamedParameterJdbcTemplate spring boot中JdbcTemplate与NamedParameterJdbcTemplate都是被自动配置的...

spring security 控制用户信息用户加密 缓存用户信息

1.MD5加密 任何一个正式的企业应用中,都不会在数据库中使用明文来保存密码的,我们在之前的章节中都是为了方便起见没有对数据库中的用户密码进行加密,这在实际应用中是极为幼稚的做法。可以想象一下,只要有人进入数据库就可以看到所有人的密码,这是一件多么恐怖的事情,为此我们至少要对密码进行加密,这样即使数据库被攻破,也可以保证用户密码的安全。 最常用的方法是使用...

Mybatis多表查询(一对一、一对多、多对多)(转)

Mybatis的多表级联查询 。 一对一可以通过实现,一对多和多对多通过实现。 元素,可以灵活选择属性column使用哪个字段进行鉴别。 一. 一对一的级联查询 对user_t表和book_t表进行连接查询。sql语句类似如下: select b.book_id,b.name,b.publishers,a.id,a.user_name from user...

mybatis常用的模糊查询

常用的模糊查询有三种方法:直接使用 % 拼接字符串,如 '%'#{name}'%' 或 "%"#{name}"%",单引号或双引号都可以。使用concat(str1,str2)函数拼接使用mybatis的bind标签<!-- ******************** 模糊查询的常用的3种方式:********************* -->...