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的生命周期
posted @ 2022-03-30 20:41  好记性不如烂笔头=>  阅读(170)  评论(0)    收藏  举报