缓存
缓存是存在内存中的临时数据,他可以使用户重复访问同一个数据时,不需要再去磁盘中再去加载,避免时间的浪费,和性能的占用。
而MyBatis中就定义了两种缓存方式:一级缓存,二级缓存。方便我们定制和查询缓存,一级缓存也就是本地缓存。二级缓存是基于namespace级别的缓存。
在默认情况下,MyBatis一般默认开启一级缓存,而二级缓存需要手动开启和配置。我们可以通过MyBatis中提供的Cache接口来自定义二级缓存。
一级缓存(本地缓存)
一级缓存是默认开启的无需我们进行配置。查询相同数据时,就会直接走一级缓存,从内存中直接调用。
一级缓存实际上是以map形式保存在内存中
一级缓存失效情况
1:sqlsession不同
sqlsession不相同会导致一级缓存失效,每个sqlsession缓存相互独立
public void testQueryUserById(){
SqlSession session = MybatisUtils.getSession();
SqlSession session2 = MybatisUtils.getSession();
UserMapper mapper = session.getMapper(UserMapper.class);
UserMapper mapper2 = session2.getMapper(UserMapper.class);
User user = mapper.selectUser(1);
System.out.println(user);
User user2 = mapper2.selectUser(1);
System.out.println(user2);
System.out.println(user==user2);
session.close();
session2.close();
//就算两个session执行的相同的sql语句,但是还是向数据库再发送了一次申请
//说明一级缓存失效
}
2:查询条件不同
当sqlsession相同时,但查询条件不同,一个为查询语句,一个增加语句,那么一级缓存还是会失效。
3:在相同的查询语句之间存在增删改等操作
在两次相同查询之间存在增删改操作。因为增删改操作会对数据库进行一个改变,所以前一个查询的缓存将不会适用于第二个查询所要查询的信息。
4:手动清除一级缓存
session.clearCache();//手动清除缓存
clearCache()方法会清空一级缓存中的信息
session.close();//关闭session也会使session中的一级缓存清除
二级缓存
因为一级缓存作用域和效率太低了,所以二级缓存应运而生。
二级缓存作用在namespace上,是为全局缓存。
如果sqlsession关闭了会话,也就是sqlsession.close();那么原本在一级缓存中的内容,就换转移到二级缓存中,方便下一次同一个mapper调用。
因为二级缓存作用于namespace,所以不同的mapper就会有不同的缓存
二级缓存的配置和开启
在MyBatis的核心配置文件中设置开启全局缓存
<setting name="cacheEnabled" value="true"/>
然后在mapper.xml中开启二级缓存
<cache/>
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<cache/>
<!-- 这就开启了二级缓存,当然二级缓存也可以进行详细配置-->
<mapper namespace="com.nicht.mapper.UserMapper">
<select id="getUser" resultype="User">
select * from User
</select>
</mapper>
二级缓存配置:
<cache
eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
这个更高级的配置创建了一个 FIFO 缓存,每隔 60 秒刷新,
最多可以存储结果对象或列表的 512 个引用,
而且返回的对象被认为是只读的,
因此对它们进行修改可能会在不同线程中的调用者产生冲突。
查到的数据最先会放在一级缓存中,只有当会话提交或者关闭时,一级缓存中的数据才会转到二级缓存中
程序最先会去一级缓存中取数据,如果一级缓存中没有数据,那么才会到二级缓存中寻找
第三方缓存实现EhCache
也可以借助第三方缓存,来更好的配置和控制缓存