目的:提高查询效率。
- 映射语句文件中的所有 select 语句的结果将会被缓存。
- 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。
- 缓存会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。
- 缓存不会定时进行刷新(也就是说,没有刷新间隔)。
- 缓存会保存列表或对象(无论查询方法返回哪种)的 1024 个引用。
- 缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。
案例分析
一级缓存:测试在一个Sesion中查询两次相同记录。
一级缓存:增删改使一级缓存失效
俩次查找id1 中间updateid2,最终导致第二次查找id1仍然查找了数据库!!!
二级缓存:关闭会话,实现一级缓存数据存储到二级缓存
SqlSession sqlSession = sqlSessionFactory.openSession(true);
SqlSession sqlSession2 = sqlSessionFactory.openSession(true);
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class);
User user = userMapper.queryUserById(1);
System.out.println(user);
sqlSession.close();
System.out.println("=======================");
User user2 = userMapper2.queryUserById(1);
System.out.println(user2);
System.out.println(user == user2);
sqlSession2.close();
一级与二级缓存对比
一级缓存:
- MyBatis的一级缓存默认开启,且默认作用范围为SESSION,即一级缓存在一个会话中生效,也可以通过配置将作用范围设置为STATEMENT,让一级缓存仅针对当前执行的SQL语句生效;
- 作用范围:sqlSession级别的缓存 (也就是拿到连接和关闭连接这个区间)
- 在同一个会话中,执行增,删,改操作会使本会话中的一级缓存失效;
- 不同会话持有不同的一级缓存,本会话内的操作不会影响其它会话内的一级缓存。
- 一般用于同一个用户在一次会话sqlSession中多次查找,可以提高查找效率。
二级缓存:
- 二级缓存需手动开启配置,可mybatis-config.xml中的<settings>中添加<setting name="cacheEnabled" value="true"/>开启全局缓存;然后在SQL映射文件添加标签<cache/>配置参数如下:
-
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
- 作用范围:namespace级别的缓存。同一命名空间(namespace)下的多个会话共享。即不同会话使用同一映射文件中的SQL语句对数据库执行操作并提交事务后,均会影响这个映射文件持有的二级缓存;
- MyBatis中执行查询操作后,需提交事务才能将查询结果缓存到二级缓存中;也就是当会话关闭后,一级缓存中的数据会被保存到二级缓存中。
- MyBatis中执行增,删或改操作并提交事务后,会清空对应的二级缓存;
- MyBatis中需要在映射文件中添加<cache>标签来为映射文件配置二级缓存,也可以在映射文件中添加<cache-ref>标签来引用其它映射文件的二级缓存以达到多个映射文件持有同一份二级缓存的效果。
小结:
同一个Mapper,对应的是操作数据库内一个表的CRUD。
一级缓存对应的就是存储一个接口内的方法操作后的数据!!!(session)
二级缓存对应的就是存储一个表内各种方法操作后的数据!!!(namespace)
所有数据都会先放到一级缓存;
只要开启了二级缓存,在同一个Mapper下就有效;
只有当会话提交或者关闭,一级缓存中数据才会保存到二级缓存中。
先直接查找二级缓存有没有数据,没有再去一级缓存找,一级缓存没有再去查找数据库,找到返回并把数据存储到一级缓存。