简介
- 什么是缓存?
- 缓存是存在于内存中的临时数据。
- 为什么使用缓存?
- 缓存能够减少和数据库的交互次数,提高执行效率。
- 什么样的数据适用于缓存,什么样的数据不适用于缓存?
- 适用于缓存:
- 经常查询,并且不经常改变的数据。
- 数据的正确与否对最终结果的影响不大。
- 不适用于缓存:
- 经常改变的数据。
- 数据的正确与否对最终结果的影响非常大。
- 适用于缓存:
- 像大多数的持久化框架一样,Mybatis 也提供了缓存策略,通过缓存策略来减少数据库的查询次数,从而提
高性能。Mybatis 中缓存分为一级缓存,二级缓存。- 一级缓存
- 当我们执行查询操作之后,Mybatis 框架会将在数据库中的查询的结果自动存储到 SqlSession 中为一级缓存划分的一块区域中。(该区域的低层结构是 Map集合)
- 当我们再次执行同样的查询操作之后,Mybatis 框架会先从 SqlSession 中的一级缓存区域查询。如果存在数据,就直接拿出来;如果不存在数据,才会去数据库中进行查询。
- 因为一级缓存区域被设置在 SqlSession 中,所以当 SqlSession 消失时,一级缓存也就消失。
- 二级缓存
- 二级缓存被存储在 SqlSessionFactory 中,同一个 SqlSessionFactory 创建的 SqlSession 共享二级缓存。
- 一级缓存
![](11.缓存\一级缓存 二级缓存.png)
目录结构
src
- main
- java
- cn.water.dao
- UserDao.java(持久层接口)
- cn.water.domain
- User.java(实体类)
- cn.water.dao
- resources
- cn.water.dao
- UserDao.xml(映射配置文件)
- SqlMapConfig.xml(MyBatis主配置文件)
- jdbcConfig.properties(数据库连接信息文件)
- cn.water.dao
- java
- test
- java.cn.water
- MybatisTest.java(测试类)
- java.cn.water
MyBatis配置文件
jdbcConfig.properties
1 | jdbc.driver=com.mysql.jdbc.Driver |
SqlMapConfig.xml
- 开启 二级缓存
1 |
|
实体类
User.java
1 | package cn.water.domain; |
持久层接口
UserDao.java
1 | package cn.water.dao; |
映射配置文件
UserDao.xml
1 |
|
测试类
MyBatisTest.java
1 | package cn.water; |
一级缓存
测试类
1 | /** 一级缓存:同一个 SqlSession */ |
运行结果
- 一级缓存是SqlSession级别的缓存,只要SqlSession没有flush或close,它就存在。
- 我们可以发现,在测试一中虽然我们查询了两次,但最后只执行了一次数据库操作,这就是Mybatis提供给我们的一级缓存在起作用了。
- 因为一级缓存的存在,导致第二次查询id为41的记录时,并没有发出sql语句从数据库中查询数据,而是从一级缓存中查询。
![](11.缓存\控制台 证明 一级缓存.png)
一级缓存的清空
测试类
1 | /** 一级缓存:不同 SqlSession */ |
运行结果
- 一级缓存是SqlSession范围的缓存,当调用SqlSession的commit()(修改,添加,删除),close()等方法时,就会清空一级缓存。
- 第一次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,如果没有,从数据库查
询用户信息。得到用户信息,将用户信息存储到一级缓存中。如果sqlSession去执行commit操作(执行插入、更新、删除),清空SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。 - 第二次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,缓存中有,直接从缓存中获取用户信息。
![](11.缓存\控制台 证明 一级缓存02.png)
二级缓存
- 当我们在使用二级缓存时,所缓存的类一定要实现 java.io.Serializable接口,这种就可以使用序列化
方式来保存对象。
开启二级缓存
- 二级缓存的使用步骤:
- 第一步:让Mybatis框架支持二级缓存(在SqlMapConfig.xml中配置)
- 第二步:让当前的映射文件支持二级缓存(在IUserDao.xml中配置
- 第三步:让当前的操作支持二级缓存(在select标签中配置)
MyBatis主配置文件
开启二级缓存支持
cacheEnabled 默认值为 true(此配置可以省略)
1 | <!-- 配置参数 --> |
映射配置文件
设置 <cache> 标签
表示当前 mapper 映射将使用二级缓存,区分的标准为 mapper 标签的 namespace 值。
1 | <mapper namespace="cn.water.dao.UserDao"> |
<select> 标签 设置 useCache=”true” 属性
注意:针对每次查询都需要最新的数据sql,要设置成useCache=false,禁用二级缓存。
1 | <mapper namespace="cn.water.dao.UserDao"> |
测试类
- 依然执行 Test02()
1 |
|
运行结果
- 经过上面的测试,我们发现执行了两次查询,并且在执行第一次查询后,我们关闭了一级缓存,再去执行第二
次查询时,我们发现并没有对数据库发出sql语句,所以此时的数据就只能是来自于我们所说的二级缓存。
一级缓存
![](11.缓存\控制台 证明 一级缓存02.png)
一级缓存和二级缓存
![](11.缓存\控制台 证明 二级缓存.png)