什么是缓存?
存在内存中的临时数据
将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系数据库)中查询,直接从缓存中查询,提高了查询效率,解决了高并发系统的性能问题
为什么使用缓存?
减少和数据库的交互次数,减少数据库的压力,减少系统开销,提高系统效率
什么样的数据能使用缓存?
经常查询并且不经常改变的数据
Mybatis缓存
Mybatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率
Mybatis系统中默认定义了两级缓存,一级缓存和二级缓存
默认情况下,只有一级缓存开启(sqlsession级别的环迅,也称为本地缓存)
二级缓存需要手动开启和配置,他是基于namespace级别的缓存
为了提高扩展性,Mybatis定义了缓存扩展接口Cache。我们可以通过实现Cache接口来自定义二级缓存
一级缓存即SqlSession级别的缓存,在一个SqlSession被建立和关闭之前生效,一级缓存是默认开启的且无法关闭
所有的查询语句的执行都会被缓存
如以下测试代码
import org.apache.ibatis.session.SqlSession; import org.junit.Test; import com.kuang.dao.UserMapper; import com.kuang.pojo.User; import com.kuang.util.MyBatisUtil; /** * 功能描述 * * @since 2022-07-25 */ public class MyTest { @Test public void testCache() { SqlSession session = MyBatisUtil.getSqlsession(); UserMapper mapper = session.getMapper(UserMapper.class); User user = mapper.getUserById(1); System.out.println(user); System.out.println("======================"); User user1 = mapper.getUserById(1); System.out.println(user1); session.close(); } }
开启日志的情况下可以看到查询结果
Created connection 988800485. ==> Preparing: select * from `user` where id = ? ==> Parameters: 1(Integer) <== Columns: id, name, pwd <== Row: 1, 狂神, 123456 <== Total: 1 User(id=1, name=狂神, pwd=123456) ====================== User(id=1, name=狂神, pwd=123456) Closing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@3aefe5e5] Returned connection 988800485 to pool.
可以看到sql的执行仅执行了一次
session.clearCache()
代码手动清除缓存一级缓存就是一个map,查询完一个就往map中插入一条key,value
二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存
二级缓存是基于namespace级别的缓存,一个命名空间,对应一个二级缓存
工作机制’
1.一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中
2.如果当前会话关闭了,这个会话对应的一级缓存就没有了,但是会话关闭后,一级缓存中的数据会被保存到二级缓存中
3.新的会话查询信息,基于可以从二级缓存中去获取
4.不同的mapper查出来的数据会放在自己对应的缓存(map)中
1.开启全局缓存
<!-- 显示开启缓存,其实不写也是默认开启的 --> <setting name="cacheEnabled" value="true" />
2.开启二级缓存
<cache/>
或者
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true" />
测试代码
import org.apache.ibatis.session.SqlSession; import org.junit.Test; import com.kuang.dao.UserMapper; import com.kuang.pojo.User; import com.kuang.util.MyBatisUtil; /** * 功能描述 * * @since 2022-07-25 */ public class MyTest { @Test public void testCache2() { SqlSession session = MyBatisUtil.getSqlsession(); UserMapper mapper = session.getMapper(UserMapper.class); User user = mapper.getUserById(1); System.out.println(user); session.close(); SqlSession session1 = MyBatisUtil.getSqlsession(); UserMapper mapper1 = session1.getMapper(UserMapper.class); User user1 = mapper1.getUserById(1); System.out.println(user1); session.close(); System.out.println(user == user1); } }