MyBatis一级缓存的笔记及记录
程序员文章站
2022-05-29 10:18:25
一、概述: 一级缓存是MyBatis天然自带的,是默认开启且没有关闭的地方,1级缓存只能作用于查询回话中,所以也叫会话缓存; 这里举个例子: 订单表存在一对多的关系,为了尽可能的减少join的查询,进行了分阶段查询; 先查询出订单表,在根据订单的用户ID查询用户信息表,最后将数据进行整合。如果订单表 ......
- 一、概述:
一级缓存是mybatis天然自带的,是默认开启且没有关闭的地方,1级缓存只能作用于查询回话中,所以也叫会话缓存;
这里举个例子:
订单表存在一对多的关系,为了尽可能的减少join的查询,进行了分阶段查询; 先查询出订单表,在根据订单的用户id查询用户信息表,最后将数据进行整合。如果订单表中存在重复的用户id,就会出现很多没有必要的重复查询。 1级缓存解决了这个问题(1个语句被执行多次,也就是大家说的“n+1”的问题),在同一次查询回话中如果出现相同的语句及参数,就会从缓存中取出,不会直接从数据库端拉取数据。
- 二、使用条件及限制:
一级缓存又称为本地缓存,大致限制如下:
- 必须是相同的sql和参数;
- 必须是同一个会话;
- 必须是相同的mapper;
- 必须是相同的接口及方法;
- 查询前不能执行session.clearcache();
- 查询语句中间不能出现增删改的原子操作,因为“增删改”操作后会自动调用session.clearcache();这里需要注意,是否更新1级缓存是取决于你的注解或者xml原标签,也就是说如果@update("select * from test where id=#{id}")也会清空1级缓存;
- 三、实现原理:
实现了cache接口——perpetualcache类,cache变量就是一个简单的hashmap,所以一级缓存就是通过hashmap实现的;
这里需要特别注意一下:
因为是基于hashmap实现,在查询时另一个会话并发去修改查询的数据的时候,一级缓存会有效,也就是非线程安全,存在并发问题,但是一级缓存又是属于本地缓存而且基于当前会话,所以这个并发问题可以忽略,您难道在自己的代码里还要实现内部并发吗?:);
- 四、缓存调用逻辑:
- 调用mapper的执行方法(这里的mapper对象是通过java代理实现的);
- defaultsqlsession.selectlist方法开始进行处理,所有查询都会进入这里;
- cachingexecutor.query:缓存执行器开始执行获取数据处理;
- baseexecutor.query:查询缓存信息。这里注意一下,clearlocalcache方法是清空缓存调用的方法,(如果当前没有关闭,就执行清空操作,这里清空是全部清空);
- perpetualcache.getobject:获取缓存;
后续可能持续更新