mybatis缓存机制
MyBatis支持一级缓存和二级缓存,默认开启
全局配置中setting name=“cacheEnabled” value=“true”(值为false时会关闭二级缓存);
select标签中useCache=false属性同理;
增删改标签设置flushCache=true设置每次清空缓存,每次操作后一级缓存与二级缓存都会被清空;
一级缓存:(本地缓存)是SqlSession级别的缓存。在操作数据库时需要构造sqlSession对象,在对象中有一个数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的;
一级缓存的生命周期
a、MyBatis在开启一个数据库会话时,会 创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象。Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉。
b、如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用。
c、如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用。
d、SqlSession中执行了任何一个update操作(update()、delete()、insert()) ,都会清空PerpetualCache对象的数据,但是该对象可以继续使用
一级缓存失效的四种情况:
1、使用不同sqlSession
2、sqlSession相同,查询条件不同(有新查询语句,当前一级缓存中还没有这个数据)
3、sqlSession相同,两次查询之间执行了增删改操作
4、sqlSession相同,手动清除了一级缓存(clean)
二级缓存:(全局缓存)是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的基于命名空间(namespace)级别的缓存,一个namespace对应一个二级缓存,它可以提高对数据库查询的效率,以提高应用的性能。
MyBatis二级缓存内部使用HashMap实现,无法实现分布式,并且服务器重启后就没有缓存了。
MyBatis二级缓存只适用于不常进行增、删、改的数据,对数据库进行增删改操作同样会清空二级缓存。清空实质是对所查key对应value置为null,而非删除键值对。
从DB中进行select的条件是:
1)缓存中不存在这个key;
2)缓存中存在这个key对应的entry对象,但是其值为null。
因为二级缓存数据存储介质多种多样,不一定只存在内存中,有可能存在硬盘中,如果我们要再取这个缓存的话,就需要反序列化了。所以建议mybatis中的pojo都要实现Serializable接口。
工作机制:
1、一个会话,查询一条数据,这个数据就会被放在当前会话的一级缓存中;
2、如果会话关闭,一级缓存中的数据会被保存到二级缓存中,新的会话查询信息时就可以参照二级缓存中的内容;
3、不同namespace查出的数据会放在自己对应的缓存中(mapper)如之前项目中user数据放在UserMapper对应的缓存下
MyBatis的缓存机制整体设计以及二级缓存的工作模式
使用:
1、开启全局二级缓存配置: (根据官方文档说明,默认开启)
2、去mapper.xml中配置使用二级缓存:
缓存元素属性见文档mybatis文档-cache
使用redis作为第三方二级缓存:
1、pom中添加相关依赖:
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>3.0.0</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.mybatis.caches/mybatis-redis -->
<dependency>
<groupId>org.mybatis.caches</groupId>
<artifactId>mybatis-redis</artifactId>
<version>1.0.0-beta2</version>
</dependency>
2、新建并配置redis.properties
mybatis-redis会自动读取redis.properties的配置
#redis的服务器地址
host=127.0.0.1
#redis的服务端口
port=6379
#链接数据库
default.db=0
#客户端超时时间单位是毫秒
timeout=60000
#最大连接数
maxActive=300
#最大空闲数
maxIdle=100
#最小空闲数
minIdle=1
#最大建立连接等待时间
maxWait=1000
#指明是否在从池中取出连接前进行检验,如果检验失败,则从池中去除连接并尝试取出另一个
testOnBorrow=true
#当调用return Object方法时,进行有效性检查
testOnReturn=true
3、在相应mapper配置文件(xxxxmapper.xml)中设置第三方缓存
<cache type="org.mybatis.caches.redis.RedisCache" blocking="false" flushInterval="0" readOnly="true" size="1024" eviction="FIFO"/>
其中参数可自行条件,细节见官方文档