MyBatis 缓存详解
https://www.cnblogs.com/wuzhenzhao/p/11103043.html MyBatis 缓存详解
https://blog.csdn.net/GDUT_Trim/article/details/120117911 MyBatis(一):BaseExecutor与CachingExecutor

https://blog.csdn.net/ThinkWon/article/details/101293216 MyBatis核心类
一级缓存(本地缓存)
“MyBatis 一级缓存的有效期是当前 SqlSession 的生命周期。只要 SqlSession 没有关闭,在这期间无论隔多久再次查询同样的 SQL 和参数,都会命中缓存。”
“但如果中间发生了写操作(insert、update、delete)或手动调用 clearCache(),缓存会被清空。”
- Mybatis的一级缓存存放在SqlSession(会话)的生命周期,在同一个SqlSession中查询时,Mybatis会把执行的方法和参数通过算法生成缓存的键值,将键值和查询结果存入一个Map对象中。如果同一个SqlSession中执行的方法和参数完全一致,那么通过算法会生成相同的键值,当Map缓存对象中已经存在改键值时,则会返回缓存中的对象。(一个SqlSession连续两次查询 得到的是同一个java对象)
- 任何的insert update delete操作都会清空一级缓存(增删改任何记录都会清空当前SqlSession的缓存)。
-
Spring整合Mybatis的时候一级缓存的问题:(默认一级缓存是开启的)
在未开启事物的情况之下(@Transactional),每次查询,spring都会关闭旧的sqlSession而创建新的sqlSession,因此此时的一级缓存是没有启作用的
在开启事物的情况之下(@Transactional),spring使用threadLocal获取当前资源绑定同一个sqlSession,因此此时一级缓存是有效的
- 总结: 一级缓存就是mapper.aaa(c,d)。 同一mapper 同一方法 同一参数。 SqlSession连续两次查询 得到的是同一个java对象(通过反序列化得到)。
- 如何实现呢。
- 利用 ThreadLocal (当前线程变量副本)和 map 实现。 ThreadLocal<Map<Object, Object>> 把 请求方法+参数 通过指定算法生成key 放到map 中,并把map 放到ThreadLocal 中,当同 同一mapper 同一方法 同一参数 查询时,会去map中的数据。 然后把数据反序列化出来,就不用和数据库交付了。
-
https://blog.csdn.net/u013887008/article/details/80379938
-
二级缓存
- 二级缓存是用来解决一级缓存不能跨会话共享的问题的,范围是namespace 级别的,可以被多个SqlSession 共享,生命周期和应用同步(同一接口里面的同一方法 多线程共享数据)。如果你的MyBatis使用了二级缓存,并且你的Mapper和select语句也配置使用了二级缓存,那么在执行select查询的时候,MyBatis会先从二级缓存中取输入,其次才是一级缓存,即MyBatis查询数据的顺序是:二级缓存 —> 一级缓存 —> 数据库。
- 每个sqlSession都有自己的一级缓存,多个sqlSession共享二级缓存
- Mybatis二级缓存可以理解为存在SqlSessionFactory的生命周期

浙公网安备 33010602011771号