Mybatis的一级、二级缓存

如图所示:

Mybatis的缓存如图所示:

当数据没有改变,开启SQLsession使用SQL语句对数据进行一次查询时,会将数据进行缓存,当第二次查询同样的数据时,则命中缓存,不去查询数据库,加快查询耗时;

特点:

一级缓存、二级缓存都是基于本地缓存的一个名为PerpetualCache,它本质是一个Hashmap

一级缓存作用域是session级别,同一个session,调同一个SQL会进行缓存,默认是打开的

二级缓存作用域更广,是namespace和mapper的作用域,只要是同一个namespace或者查询同一个mapper文件(同一张表)的SQL,都会进行缓存,不依赖session,但是默认是关闭的;

一、一级缓存

1、基于一个session

基于perpetualCache的HashMap本地缓存,其作用域为Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存

java 复制代码
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession = sqlSessionFactory.openSession();

// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1、userMapper2两个代理对象
UserMapper userMapper1 = sqlsession.getMapper(UserMapper.class);
UserMapper userMapper1 = sqlsession.getMapper(UserMapper.class);

// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);

System.out.prientln("--------------------------");
User user1 = userMapper2.selectById(6);
System.out.prientln(user1);

控制台结果如下:

我们可以看到,明明调用了两次select语句,但是实际上只查询了一次数据库,证明后面一次直接走的是缓存,这就是一级缓存

因为他们用的是同一个session,走了一级缓存,用两个session就会查询两次数据库了

2、基于两个session

java 复制代码
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession1 = sqlSessionFactory.openSession();

// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1代理对象
UserMapper userMapper1 = sqlsession1.getMapper(UserMapper.class);

// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);
sqlsession1.close();

// 另外开启一个session
Sqlsession sqlsession2 = sqlSessionFactory.openSession();

System.out.prientln("--------------------------");
UserMapper userMapper2 = sqlsession2.getMapper(UserMapper.class);
User user2 = userMapper2.selectById(6);
System.out.prientln(user2);

sqlsession2.close();

控制台结果如下:

可以看到,以及缓存失效了,查询了两次数据库;

另外:当数据有过增删改等操作后,缓存就会失效,也会查两次数据库

那如果我两个session,也想使用Mybatis的缓存,可以不可以呢?

可以的,那就需要使用二级缓存了:

二、二级缓存

二级缓存也是基于perpetualCache的HashMap本地缓存,其作用域是namespace和mapper;而不是依赖于SQL session,所以范围更广

当两个session不能缓存时,我们可以手动配置开启二级缓存,使不同的session也能缓存:

开启方式,两步:

1、全局配置文件:

XML 复制代码
<settings>
    <setting name='cacheEnabled" value='true'
</settings>

2、mapper映射文件中:

使用<cache/>标签让当前mapper文件生效二级缓存

我们测试一下,还是上面那个一级缓存失效的例子:

java 复制代码
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession1 = sqlSessionFactory.openSession();

// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1代理对象
UserMapper userMapper1 = sqlsession1.getMapper(UserMapper.class);

// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);
sqlsession1.close();

// 另外开启一个session
Sqlsession sqlsession2 = sqlSessionFactory.openSession();

System.out.prientln("--------------------------");
UserMapper userMapper2 = sqlsession2.getMapper(UserMapper.class);
User user2 = userMapper2.selectById(6);
System.out.prientln(user2);

sqlsession2.close();

控制台结果如下:

手动二级缓存后,缓存生效

注意事项:

1、对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存namespace)的数据,进行了新增、修改、删除操作后,默认该作用域下的所有select中的缓存将被清除clear

2、二级缓存需要缓存中的数据,实现Serializable接口

3、只有会话提交或者关闭后,一级缓存中的数据才会转移到二级缓存中

相关推荐
llwszx3 小时前
深入理解Java锁原理(一):偏向锁的设计原理与性能优化
java·spring··偏向锁
云泽野3 小时前
【Java|集合类】list遍历的6种方式
java·python·list
二进制person4 小时前
Java SE--方法的使用
java·开发语言·算法
小阳拱白菜5 小时前
java异常学习
java
FrankYoou6 小时前
Jenkins 与 GitLab CI/CD 的核心对比
java·docker
麦兜*6 小时前
Spring Boot启动优化7板斧(延迟初始化、组件扫描精准打击、JVM参数调优):砍掉70%启动时间的魔鬼实践
java·jvm·spring boot·后端·spring·spring cloud·系统架构
KK溜了溜了7 小时前
JAVA-springboot 整合Redis
java·spring boot·redis
天河归来7 小时前
使用idea创建springboot单体项目
java·spring boot·intellij-idea
weixin_478689767 小时前
十大排序算法汇总
java·算法·排序算法
码荼7 小时前
学习开发之hashmap
java·python·学习·哈希算法·个人开发·小白学开发·不花钱不花时间crud