Mybatis缓存介绍
使用缓存可以避免与数据库频繁交互,从而减少数据库的压力,并且提升查询效率。Mybatis提供了一级缓存和二级缓存,并且预留了集成第三方缓存的接口。
Mybatisd的cache包内包含了与缓存相关的类,其中有一个Cache接口,它有一个默认的实现类PerpetualCache,是一个用HashMap实现的本地缓存。
一级缓存
一级缓存也叫本地缓存,Mybatis默认开启了一级缓存。一级缓存的作用域为SqlSession,就是说同一个SqlSession对象,在参数和sql完全一样的情况下,只执行一次sql语句,第一次执行完毕会将结果写入缓存中,第二次会直接从缓存中获取数据而不再到数据库中查询。
一级缓存的生命周期
- Mybatis开启一个数据库会话时,会创建一个新的SqlSession对象,SqlSession对象中有一个Executor对象持有一个新的PerpetualCache对象,这个PerpetualCache对象维持的一级缓存。
- SqlSession的close()方法被调用时,会释放PerpetualCache对象,一级缓存即不可用。
- SqlSession的clearCache()方法会清空PerpetualCache对象中的数据,此时缓存为空,但该对象仍可用。
- 当SqlSession执行了update()、delete()、insert()操作时,由于数据可能发生改变,因此也会清空PerpetualCache对象。
- 当一个SqlSession结束时,SqlSession对象及其内部的Executor对象以及PerpetualCache对象全部被释放。
一级缓存的实现
- 在一个SqlSession中,Mybatis会把执行的查询方法和参数通过算法生成缓存的key键,通过这个key键在缓存的Map中获取数据。
- 如果key键对应的value值为空,则去数据库中查询数据,并将数据作为value值存入key键对应的缓存Map中。
- 如果key键对应的value值不为空,则直接返回value值。
二级缓存
由于一级缓存是session级别的缓存,不同会话不能共享缓存,因此,不同会话之间对于相同的数据可能有不一样的缓存,会存在脏数据的问题。二级缓存存在于SqlSessionFactory生命周期中,作用范围是namespace级别的,它解决了缓存的跨会话共享的问题。在执行查询时,Mybatis会先从二级缓存中取值,其次才是一级缓存,都未命中才去数据库查询。
二级缓存的配置
二级缓存需要配置。由于增删改操作会刷新缓存,因此在查询为主的应用中配置二级缓存才有意义。
- Mybatis中二级缓存有一个全局开关,即配置cacheEnabled,默认为true。
- 在全局配置开启的情况下,在单个mapper.xml中添加
<cache/>
标签即开启了此mapper的namespace下的二级缓存。
二级缓存的特点
- Mybatis中的二级缓存作用于mapper范围级别
- 二级缓存是事务性的,事务提交后才会将数据写入二级缓存
- 增删改操作会清空缓存