二级缓存
-
二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存
-
基于namespace级别的缓存,一个名称空间,对应一个二级缓存;
-
工作机制
- 一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中;
- 如果当前会话关闭了,这个会话对应的一级缓存就没了;但是我们想要的是,会话关闭了,一级缓存中的数据被保存到二级缓存中;
- 新的会话查询信息。就可以从二级缓存中获取内容;
- 不同的mapper查出的数据会放在自己对应的缓存(map)中;
步骤:
1.开启全局缓存
<settings>
<setting name="logImpl" value="STDOUT_LOGGING"/>
<!-- 开启全局缓存 -->
<setting name="cacheEnabled" value="true"/>
</settings>
2.要在使用二级缓存的Mapper中开启
<!--在当前Mapper.xml中使用二级缓存-->
<cache />
也可以自定义参数
<!--在当前Mapper.xml中使用二级缓存-->
<cache eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"
/>
3.测试
- 问题:我们需要将实体类序列化!否则就会报错
caused by: java.io.NotserializableException: com.tian.pojo.user
//解决方法
实体类 implements serializable
报错:Cause: org.apache.ibatis.executor.ExecutorException: Executor was closed.
还未解决
@Test
public void test(){
SqlSession sqlSession = MybatisUtils.getSqlSession();
SqlSession sqlSession2 = MybatisUtils.getSqlSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
User user1 = userMapper.queryUserById(1);
System.out.println(user1);
sqlSession.close();
UserMapper userMapper2 = sqlSession.getMapper(UserMapper.class);
User user2 = userMapper2.queryUserById(1);
System.out.println(user2);
System.out.println(user1==user2);
sqlSession2.close();
}
4.运行结果
Opening JDBC Connection
Created connection 664792509.
Setting autocommit to false on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@279fedbd]
==> Preparing: select * from user where id=?
==> Parameters: 1(Integer)
<== Columns: id, name, pwd
<== Row: 1, 狂神, 123456
<== Total: 1
User(id=1, name=狂神, pwd=123456)
Cache Hit Ratio [com.tian.dao.UserMapper]: 0.0
User(id=1, name=狂神, pwd=123456)
true
Resetting autocommit to true on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@279fedbd]
Closing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@279fedbd]
Returned connection 664792509 to pool.
小结:
- 只要开启了二级缓存,在同一个Mapper下就有效
- 所有的数据都会先放在一级缓存中;
- 只有当会话提交,或者关闭的时候。才会提交到二级缓冲中!