十三、缓存
1. 简介
查询-->需要连接数据库,耗资源 一次查询的结果,给他暂存一个可以直接取到的地方-->内存:缓存 我们再次查询相同数据的时候,直接走缓存,就不用走数据库了
1.什么是缓存[Cache]?
- 存在内存中的临时数据
- 将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题。
2.为什么要使用缓存?
- 减少和数据库的交互次数,减少系统开销,提高系统效率
3.什么样的数据能使用缓存?
- 经常查询并且不经常改变的数据
2. Mybatis缓存
1.Mybatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大地提升查询效率。
2.Mybatis系统中默认定义了两级缓存:一级缓存和二级缓存
- 默认情况下,只有一级缓存开启。(Sqlsession级别的缓存,也称为本地缓存)
- 二级缓存需要手动开启和配置,是基于namespace级别的缓存
- 为了提高扩展性,mybatis定义了缓存接口,我们可以通过实现Cache接口来自定义二级缓存。
3. 一级缓存
- 一级缓存也叫本地缓存:SqlSession
- 与数据库同一次会话期间查询到得数据会放在本地缓存中
- 以后如果需要获取相同得数据,直接从换从中拿,没必要再去查询数据库
3.1 测试步骤:
1. 开启日志
在核心配置文件中
<settings> <!--标准的日志工厂实现--> <setting name="logImpl" value="STDOUT_LOGGING"/> </settings>
2.测试再一个Session中查询两次相同得记录
@Test public void queryUserByIdTest(){ 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); sqlSession.close(); }
3.查看日志输出
3.2 缓存失效得情况
- 查询不同得东西
- 增删改操作可能会改变原来得数据,所以必定会刷新缓存
- 比如说在查两个相同的用户,中间插一个修改用户的操作
- 查询不同Mapper.xml
- 手动清理缓存
sqlSession.clearCache();//手动清理缓存
3.3 小结
一级缓存默认是开启的,只在一次sqlSession中有效,也就是拿到连接到关闭连接这个区间
一级缓存就相当于一个map
4. 二级缓存
4.1 简介
4.2 开启步骤
1.开启全局缓存
虽然默认就是true但是还是需要显示的开启一下,增强可读性
在核心配置文件中
<settings> <!--显示的开启全局缓存--> <setting name="cacheEnabled" value="true"/> </settings>
2.要启用二级缓存需要在sql映射文件中添加一行代码
<!--在当前Mapper.xml中开启二级缓存--> <cache/>
也可以自定义一些参数
<!--官网上是这个。加上会有一些设定,比如先进先出,60秒,512个引用,只读为true--> <cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
还可以在你想要的地方关闭缓存,只需要在sql语句标签下添加下面的代码
4.3 测试
1. 如果没有加二级缓存,查询两个sqlSession是会进入两次数据库的
@Test public void queryUserByIdTest(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); SqlSession sqlSession2 = MybatisUtils.getSqlSession(); UserMapper mapper = sqlSession.getMapper(UserMapper.class); UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class); User user = mapper.queryUserById(1); System.out.println(user); User user1 = mapper2.queryUserById(1); System.out.println(user1); sqlSession.close(); sqlSession2.close(); }
2.如果开启了二级缓存
@Test public void queryUserByIdTest(){ SqlSession sqlSession = MybatisUtils.getSqlSession(); SqlSession sqlSession2 = MybatisUtils.getSqlSession(); UserMapper mapper = sqlSession.getMapper(UserMapper.class); User user = mapper.queryUserById(1); System.out.println(user); sqlSession.close(); UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class); User user1 = mapper2.queryUserById(1); System.out.println(user1); System.out.println(user == user1); sqlSession2.close(); }
3.问题:
我们需要将实体类序列化,否则就会报错
Caused by:java.io.NotSerializableException:com.hxl.pojo.User
我们需要在实体类后加上implements Serializable
@Data public class User implements Serializable { private int id; private String name; private String pwd; }
4.4 小结
- 只要开启了二级缓存,在同一个Mapper下就有效
- 所有的数据都会先放在一级缓存中
- 只有当会话提交,或者关闭的时候,才会提交到二级缓存中
- 实体类要序列化
5. 缓存原理
6. 自定义缓存-Ehcache
Ehcache是一种广泛使用的开源Java分布式缓存,主要面向通用缓存
要在程序中使用Ehcache,先要导包
<dependency> <groupId>org.mybatis.caches</groupId> <artifactId>mybatis-ehcache</artifactId> <version>1.2.1</version> </dependency>
在resources下建立一个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"> <!-- --> <diskStore path="./tmpdir/Tmp_EhCache"/> <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"/> </ehcache>
在需要的Mapper.xml下
<cache type="org.mybatis.caches.ehcache.EhcacheCache"></cache>
测试