欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

MyBatis一级缓存的笔记及记录

程序员文章站 2023-04-05 08:17:28
一、概述: 一级缓存是MyBatis天然自带的,是默认开启且没有关闭的地方,1级缓存只能作用于查询回话中,所以也叫会话缓存; 这里举个例子: 订单表存在一对多的关系,为了尽可能的减少join的查询,进行了分阶段查询; 先查询出订单表,在根据订单的用户ID查询用户信息表,最后将数据进行整合。如果订单表 ......
  • 一、概述:

   一级缓存是mybatis天然自带的,是默认开启且没有关闭的地方,1级缓存只能作用于查询回话中,所以也叫会话缓存;

这里举个例子:

  订单表存在一对多的关系,为了尽可能的减少join的查询,进行了分阶段查询; 先查询出订单表,在根据订单的用户id查询用户信息表,最后将数据进行整合。如果订单表中存在重复的用户id,就会出现很多没有必要的重复查询。 1级缓存解决了这个问题(1个语句被执行多次,也就是大家说的“n+1”的问题),在同一次查询回话中如果出现相同的语句及参数,就会从缓存中取出,不会直接从数据库端拉取数据。

  • 二、使用条件及限制:

一级缓存又称为本地缓存,大致限制如下:

  1. 必须是相同的sql和参数;
  2. 必须是同一个会话;
  3. 必须是相同的mapper;
  4. 必须是相同的接口及方法;
  5. 查询前不能执行session.clearcache();
  6. 查询语句中间不能出现增删改的原子操作,因为“增删改”操作后会自动调用session.clearcache();这里需要注意,是否更新1级缓存是取决于你的注解或者xml原标签,也就是说如果@update("select * from test where id=#{id}")也会清空1级缓存;
  • 三、实现原理:

  实现了cache接口——perpetualcache类,cache变量就是一个简单的hashmap,所以一级缓存就是通过hashmap实现的;

这里需要特别注意一下:

  因为是基于hashmap实现,在查询时另一个会话并发去修改查询的数据的时候,一级缓存会有效,也就是非线程安全,存在并发问题,但是一级缓存又是属于本地缓存而且基于当前会话,所以这个并发问题可以忽略,您难道在自己的代码里还要实现内部并发吗?:);

  • 四、缓存调用逻辑:
  1. 调用mapper的执行方法(这里的mapper对象是通过java代理实现的);
  2. defaultsqlsession.selectlist方法开始进行处理,所有查询都会进入这里;
  3. cachingexecutor.query:缓存执行器开始执行获取数据处理;
  4. baseexecutor.query:查询缓存信息。这里注意一下,clearlocalcache方法是清空缓存调用的方法,(如果当前没有关闭,就执行清空操作,这里清空是全部清空);
  5. perpetualcache.getobject:获取缓存;

后续可能持续更新