1.缓存
1.1 简介
查询: 连接数据库,耗资源!
一次查询的结果,给他暂存在一个可以直接取到的地方!—>内存:缓存
我们再次查询相同数据的时候,直接缓存,就不用走数据库了
- 什么是缓存【Cache】
- 存在内存中的临时数据。
- 将用户经常访问的数据放到缓存(内存)中
- 为什么使用缓存
- 减少和数据库的交互次数,减少系统开销,提高系统效率。
- 什么样的数据能使用缓存?
- 经常查询并不经常改变的数据。【可以使用缓存】
1.2 Mybatis默认定义了两级缓存:
一级缓存
-
一级缓存默认开启,只在一次sqlSession中有效,也就是从getSqlSession()到close()的过程之间
-
两次查询只执行了一次sql语句,第一次次sql查询被加入到缓存当中。
接口:
User queryUserById(@Param("id") int id);
mapper.xml
<select id="queryUserById" resultType="user" parameterType="_int"> select * from mybatis_test.m_user where id = #{id} </select>
测试类:
@Test public void test(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); UserMapper mapper = sqlSession.getMapper(UserMapper.class); User user = mapper.queryUserById(1); System.out.println(user); System.out.println("-----------------------"); User user1 = mapper.queryUserById(1); System.out.println(user1); System.out.println(user==user1); sqlSession.close(); }
结果:
-
缓存失效的情况:
update情况:
1. 查询不同的东西,如不同的id 2.增删改语句就会可能改变原来的数据内容。因此,执行完增删改之后,就会重新执行sql,刷新缓存。 3.查询不同的Mapper.xml 4.手动清理缓存:sqlSession.clearCache(); 小结:一级缓存默认开启,只在一次sqlSession中有效,也就是从getSqlSession()到close()的过程之间
一级缓存相当于Map,用过就没了
二级缓存
1. 步骤:
-
开启全局缓存,默认是开启的,但是要主动声明一下。在mybatis文档是(setting配置)中。namespace级别的缓存
cacheEnabled 全局地开启或关闭配置文件中的所有映射器已经配置的任何缓存。 默认 true -
显示开启全局缓存
<setting name="cacheEnabled" value="true"/>
-
我们需要对实体序列化,否则就会报错
-
public class User implements Serializable {
}
- 在UserMapper.xml中配置缓存,namespace级别的缓存代码:
```java
<!-- 配置二级缓存。以IO流为格式,每隔60s缓存一次,最多缓存512个,是否只读,(此些配置完全可以不写),pojo加@AllArgsConstructor注解并实现序列化-->
<cache
eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
<!-- useCache=false表示这条语句不进入缓存,true是进入缓存 -->
<select id="quaryUserById" resultType="user" useCache="true">
select * from mybatis.user where id = #{id}
</select>
2. 查询顺序:
- 先找二级缓存,再找一级缓存,如果都没有,就去连接数据库
小结:
-
只要开启了二级缓存,在同一个Mapper下就会生效
-
所有的数据都会先放在一级缓存中
-
只有当会话提交,或者关闭,才会提交到二级缓存
13.3 自定义缓存(Ehcache)
Ehcache是一种广泛使用的开源Java分布式缓存,主要面向通用缓存
步骤:
-
导入依赖:
<!-- https://mvnrepository.com/artifact/org.mybatis.caches/mybatis-ehcache --> <dependency> <groupId>org.mybatis.caches</groupId> <artifactId>mybatis-ehcache</artifactId> <version>1.1.0</version> </dependency>
-
在接口实现Mapper.xml中:
<cache type="org.mybatis.caches.encache.EhcacheCache"/>
-
在resource中创建ehcache.xml文件:
<?xml version="1.0" encoding="UTF-8"?> <ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd" updateCheck="false"> <defaultCache eternal="false" maxElementsInMemory="10000" overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="1800" timeToLiveSeconds="259200" memoryStoreEvictionPolicy="LRU"/> <cache name="cloud_user" eternal="false" maxElementsInMemory="5000" overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="1800" timeToLiveSeconds="1800" memoryStoreEvictionPolicy="LRU"/> <!-- diskStore:为缓存路径,ehcache分为内存和磁盘两级,此属性定义磁盘的缓存位置。参数解释如下: user.home – 用户主目录 user.dir – 用户当前工作目录 java.io.tmpdir – 默认临时文件路径 --> <diskStore path="java.io.tmpdir/Tmp_EhCache"/> <!-- defaultCache:默认缓存策略,当ehcache找不到定义的缓存时,则使用这个缓存策略。只能定义一个。 --> <!-- name:缓存名称。 maxElementsInMemory:缓存最大数目 maxElementsOnDisk:硬盘最大缓存个数。 eternal:对象是否永久有效,一但设置了,timeout将不起作用。 overflowToDisk:是否保存到磁盘,当系统当机时 timeToIdleSeconds:设置对象在失效前的允许闲置时间(单位:秒)。仅当eternal=false对象不是永久有效时使用,可选属性,默认值是0,也就是可闲置时间无穷大。 timeToLiveSeconds:设置对象在失效前允许存活时间(单位:秒)。最大时间介于创建时间和失效时间之间。仅当eternal=false对象不是永久有效时使用,默认是0.,也就是对象存活时间无穷大。 diskPersistent:是否缓存虚拟机重启期数据 Whether the disk store persists between restarts of the Virtual Machine. The default value is false. diskSpoolBufferSizeMB:这个参数设置DiskStore(磁盘缓存)的缓存区大小。默认是30MB。每个Cache都应该有自己的一个缓冲区。 diskExpiryThreadIntervalSeconds:磁盘失效线程运行时间间隔,默认是120秒。 memoryStoreEvictionPolicy:当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理内存。默认策略是LRU(最近最少使用)。你可以设置为FIFO(先进先出)或是LFU(较少使用)。 clearOnFlush:内存数量最大时是否清除。 memoryStoreEvictionPolicy:可选策略有:LRU(最近最少使用,默认策略)、FIFO(先进先出)、LFU(最少访问次数)。 FIFO,first in first out,这个是大家最熟的,先进先出。 LFU, Less Frequently Used,就是上面例子中使用的策略,直白一点就是讲一直以来最少被使用的。如上面所讲,缓存的元素有一个hit属性,hit值最小的将会被清出缓存。 LRU,Least Recently Used,最近最少使用的,缓存的元素有一个时间戳,当缓存容量满了,而又需要腾出地方来缓存新的元素的时候,那么现有缓存元素中时间戳离当前时间最远的元素将被清出缓存。 --> </ehcache>
现在我们使用Redis数据库做缓存!
欢迎访问我的个人博客:http://www.ayjup.cn