Mybatis缓存
正如大多数持久层框架一样,mybatis同样提供了以及缓存和二级缓存的支持
一级缓存(默认存在):
基于PerpetualCache 的 HashMap本地缓存(mybatis内部实现cache接口), 其存储作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 Cache 就将清空;
mybatis 默认提供以及缓存,缓存的范围是一个 sqlSession,在同一个SqlSession中,执行两次相同的sql查询,第二次不会再数据库中查询,而是在缓存中拿数据
原理; 以及缓存采用的是Hashmap储存,mybatis执行查询是,从缓存中查询,如果缓存中没有的话,再从数据库中进行查询>
如果SqlSession执行的是 clearCache() 提交或者增加 删除 修改等操作,清除缓存
二级缓存:
一级缓存是在同一个sqlSession中,二级缓存是在同一个namespace中,因此相同的的namespace不同的sqlSession可以使用二级缓存
使用场景:
1、对查询频率高,变化频率低的数据建议使用二级缓存。
2、对于访问多的查询请求且用户对查询结果实时性要求不高,此时可采用
mybatis 二级缓存技术降低数据库访问量,提高访问速度,业务场景比
如:耗时较高的统计分析 sql、电话账单查询 sql 等。
二级缓存的配置
全局文件配置:
使用二级缓存
<setting name="cacheEnabled" value="true"/>
开启该mapper的二级缓存 :`
cache标签常用属性
<cache
eviction="FIFO" <!--回收策略为先进先出-->
flushInterval="60000" <!--自动刷新时间 60s-->
size="512" <!--最多缓存 512 个引用对象-->
readOnly="true"/> <!--只读-->
说明:
-
映射语句文件中的所有 select 语句将会被缓存。
-
映射语句文件中的所有 insert,update 和 delete 语句会刷新缓存。
-
缓存会使用 Least Recently Used(LRU,最近最少使用的)算法来收回。
-
缓存会根据指定的时间间隔来刷新.
-
缓存会存储 1024 个对象
po对象必须支持实例化:
实现Serializable接口
public class User implements Serializable {
}
如果关闭Mapper下具体的statement 缓存?
正常使用useCache: 默认是true
如果要关闭把useCache 改为false就可以
<select id="findUserByid" parameterType="int" resultType="User" useCache="false">
SELECT * FROM user WHERE id=#{id}
</select>
刷新二级缓存:
操作 CUD的 statement时候,会强制刷新二级缓存 即默认 flushCache=“true” ,如果想关 闭设定为 flushCache="false"即可 ,不建议关闭刷新,因为操作更新删除修改,关闭后容 易获取脏数据
分布式缓存 ehcache:
如果有多台服务器 ,不使用分布缓存,缓存的数据在各个服务器单独存储,不方便 系统 开发。
所以要使用分布式缓存对缓存数据进行集中管理。因此可是使用 ehcache memcached redis
mybatis本身来说是无法实现分布式缓存的,所以要与分布式缓存框架进行整合。
EhCache 是一个纯 Java 的进程内缓存框架,具有快速、精干等特点;
Ehcache 是一种广泛 使用的开源 Java分布式缓存。
主要面向通用缓存,Java EE和轻量级容器。
它具有内存和磁盘存储,缓存加载器,缓存扩展,缓存异常处理程序,
一个 gzip 缓存 servlet过滤器,支持 REST和 SOAP api等特点。
jar依赖:
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.7.12</version>
</dependency>
<dependency>
<groupId>net.sf.ehcache</groupId>
<artifactId>ehcache-core</artifactId>
<version>2.4.4</version>
</dependency>
<dependency>
<groupId>org.mybatis.caches</groupId>
<artifactId>mybatis-ehcache</artifactId>
<version>1.0.3</version>
</dependency
缓存接口配置:
在 resources下 加入 ehcache.xml
<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="../bin/ehcache.xsd">
<!-name:Cache 的唯一标识
maxElementsInMemory:内存中最大缓存对象数
maxElementsOnDisk:磁盘中最大缓存对象数,若是 0 表示无穷大
eternal:Element 是否永远不过期,如果为 true,则缓存的数据始终有效,如果为 false
那么还要根据 timeToIdleSeconds,timeToLiveSeconds 判断
overflowToDisk:配置此属性,当内存中 Element 数量达到 maxElementsInMemory 时,
Ehcache 将会 Element 写到磁盘中
timeToIdleSeconds:设置 Element 在失效前的允许闲置时间。仅当 element 不是永久有效
时使用,可选属性,默认值是 0,也就是可闲置时间无穷大
timeToLiveSeconds:设置 Element 在失效前允许存活时间。最大时间介于创建时间和失效
时间之间。仅当 element 不是永久有效时使用,默认是 0.,也就是 element 存活时间无穷
大
diskPersistent:是否缓存虚拟机重启期数据
diskExpiryThreadIntervalSeconds:磁盘失效线程运行时间间隔,默认是 120 秒
diskSpoolBufferSizeMB:这个参数设置 DiskStore(磁盘缓存)的缓存区大小。默认是
30MB。每个 Cache 都应该有自己的一个缓冲区
memoryStoreEvictionPolicy:当达到 maxElementsInMemory 限制时,Ehcache 将会根据
指定的策略去清理内存。默认策略是 LRU(最近最少使用)。你可以设置为 FIFO(先进先
出)或是 LFU(较少使用)
-->
<defaultCache overflowToDisk="true" eternal="false"/>
<diskStore path="D:/cache" />
<!-
<cache name="sxtcache" overflowToDisk="true" eternal="false"
timeToIdleSeconds="300" timeToLiveSeconds="600" maxElementsInMemory="1000"
maxElementsOnDisk="10" diskPersistent="true"
diskExpiryThreadIntervalSeconds="300"
diskSpoolBufferSizeMB="100" memoryStoreEvictionPolicy="LRU" />
-->