简介
所有的查询都要连接数据库, 比较耗资源
一次查询的结果,给它暂存在一个可以直接取到的地方 –>内存:缓存
我们再次查询相同数据的时候, 直接走缓存, 就不用走数据库了
什么是缓存[Cache]?
- 存在内存中的临时数据
- 将用户经常查询的数据放在缓存(内存)中, 用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询, 从缓存中查询, 从而提高查询效率, 解决了高并发系统的性能问题
为什么使用缓存?
- 减少和数据库的交互次数, 减少系统开销, 提高系统效率
什么样的数据能使用缓存?
- 经常查询并且不经常改变的数据
Mybatis缓存
Mybatis包含一个非常强大的查询缓存特性, 它可以非常方便地定制和配置缓存. 缓存可以极大地提升查询效率
Mybatis系统中默认定义了两级缓存:一级缓存 和 *二级缓存 *
- 默认情况下, 只有一级缓存开启(SqlSession级别的缓存, 也称为本地缓存)
- 二级缓存需要手动开启和配置, 他是基于namespace级别的缓存
- 为了提高扩展性, Mybatis定义了缓存接口Cache, 我们可以通过实现Cache接口来自定义二级缓存
默认情况下,只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存。 要启用全局的二级缓存,只需要在你的 SQL 映射文件中添加一行:
1 | <cache/> |
提示 缓存只作用于 cache 标签所在的映射文件中的语句。如果你混合使用 Java API 和 XML 映射文件,在共用接口中的语句将不会被默认缓存。你需要使用 @CacheNamespaceRef 注解指定缓存作用域。
这些属性可以通过 cache 元素的属性来修改。比如:
1 | <cache |
这个更高级的配置创建了一个 FIFO 缓存,每隔 60 秒刷新,最多可以存储结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此对它们进行修改可能会在不同线程中的调用者产生冲突。
可用的清除策略有:
LRU
– 最近最少使用:移除最长时间不被使用的对象。FIFO
– 先进先出:按对象进入缓存的顺序来移除它们。SOFT
– 软引用:基于垃圾回收器状态和软引用规则移除对象。WEAK
– 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。
默认的清除策略是 LRU。
flushInterval(刷新间隔)属性可以被设置为任意的正整数,设置的值应该是一个以毫秒为单位的合理时间量。 默认情况是不设置,也就是没有刷新间隔,缓存仅仅会在调用语句时刷新。
size(引用数目)属性可以被设置为任意正整数,要注意欲缓存对象的大小和运行环境中可用的内存资源。默认值是 1024。
readOnly(只读)属性可以被设置为 true 或 false。只读的缓存会给所有调用者返回缓存对象的相同实例。 因此这些对象不能被修改。这就提供了可观的性能提升。而可读写的缓存会(通过序列化)返回缓存对象的拷贝。 速度上会慢一些,但是更安全,因此默认值是 false。
一级缓存
一级缓存也叫本地缓存: SqlSession
- 与数据库同一次会话期间查询到的数据会放在本地缓存里
- 以后如果需要获取相同的数据, 直接从缓存中拿, 没必要再去查询数据库
- 默认开启! 也关闭不掉, 从拿到连接到关闭连接
测试步骤:
开启日志!
搭建一个项目, 完成简单的查询功能
1
2
3
4
5<mapper namespace="com.lxb.dao.UserMapper">
<select id="queryUserById" parameterType="_int" resultType="user">
select * from mybatis.user where id=#{id}
</select>
</mapper>在测试时对一个对象查询两次:
1
2
3
4
5
6
7
8
9
10
11
12
13
public void test() {
SqlSession sqlSession = MybtisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.queryUserById(1);
System.out.println(user);
System.out.println("========================");
User user2 = mapper.queryUserById(1);
System.out.println(user2);
System.out.println(user == user2);
sqlSession.close();
}输出结果为:
1
2
3
4
5
6
7
8
9
10
11
12
13Opening JDBC Connection
Created connection 8805846.
==> Preparing: select * from mybatis.user where id=?
==> Parameters: 1(Integer)
<== Columns: id, name, pwd
<== Row: 1, 刘潇博, 123456
<== Total: 1
User{id=1, name='刘潇博', password='123456'}
========================
User{id=1, name='刘潇博', password='123456'}
true
Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@865dd6]
Returned connection 8805846 to pool.分析日志可知: 两次的查询操作都是在同一个会话中完成的, 两次取得的对象相同
缓存失效的情况
- 查询不同的东西
- 增删改操作, 可能会改变原来的数据, 所以必定会刷新缓存
- 查询不同的Mapper.xml
- 手动清理缓存
手动清理缓存语句:
1 | sqlSession.clearCache(); |
二级缓存
二级缓存也叫全局缓存, 一级缓存作用域太低了, 所以诞生了二级缓存
基于namespace级别的缓存, 一个名称空间, 对应一个二级缓存
- 工作机制
- 一个会话查询一条数据, 这个数据就会被放在当前会话的一级缓存中
- 如果当前会话关闭了, 这个会话对应的一级缓存就没了, 但是我们想要的是, 会话关闭了, 一级缓存中的数据被保存到二级缓存中
- 新的会话查询信息, 就可以直接从二级缓存中获取内容
- 不同的Mapper值查出的数据会放在自己对应的缓存中
开启全局缓存
1
<setting name="cacheEnabled" value="true"/>
默认也会开启, 但是写出来可读性好
在要使用二级缓存的Mapper中开启, 在具体的SQL语句标签中使用该缓存
1
2
3
4
5
6
7
8
9
10<!--在当前Mapper.xml中使用二级缓存-->
<cache
eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
<select id="queryUserById" parameterType="_int" resultType="user" useCache="true">
select * from mybatis.user where id=#{id}
</select>也可以不自定义缓存属性, 直接写个空的
1
<cache/>
测试代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public void test() {
SqlSession sqlSession = MybtisUtils.getSqlSession();
SqlSession sqlSession1 = MybtisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.queryUserById(2);
System.out.println(user);
sqlSession.close();
UserMapper mapper1 = sqlSession1.getMapper(UserMapper.class);
User user1 = mapper1.queryUserById(2);
System.out.println(user1);
System.out.println("==============");
System.out.println(user==user1);
sqlSession1.close();
}注意要把第一个会话关闭之后,再开启新的会话
结果
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15Cache Hit Ratio [com.lxb.dao.UserMapper]: 0.0
Opening JDBC Connection
Created connection 1804379080.
==> Preparing: select * from mybatis.user where id=?
==> Parameters: 2(Integer)
<== Columns: id, name, pwd
<== Row: 2, asfsdaf, 98864
<== Total: 1
User{id=2, name='asfsdaf', password='98864'}
Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@6b8ca3c8]
Returned connection 1804379080 to pool.
Cache Hit Ratio [com.lxb.dao.UserMapper]: 0.5
User{id=2, name='asfsdaf', password='98864'}
==============
true可以看出第二个会话直接从缓存中取结果
小结:
- 只要开启了二级缓存, 在同一个Mapper下就有效
- 所有的数据都会先放在一级缓存中, 只有当会话提交或者关闭的时候才会转存到二级缓存中
自定义缓存-Ehcache
Ehcache是一种广泛使用的开源Java分布式缓存, 主要面向通用缓存
要在程序中使用, 先导包
1 | <!-- https://mvnrepository.com/artifact/org.mybatis.caches/mybatis-ehcache --> |
在Mapper中指定使用的自定义缓存类型
1 | <!-- 在当前Mapper.xml中使用二级缓存--> |
创建针对于Ehcache的配置文件
1 |
|