当前位置: 首页 > article >正文

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就将清空,默认打开一级缓存

// 获取一个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

// 获取一个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、全局配置文件:

<settings>
    <setting name='cacheEnabled" value='true'
</settings>

2、mapper映射文件中:

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

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

// 获取一个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、只有会话提交或者关闭后,一级缓存中的数据才会转移到二级缓存中


http://www.kler.cn/a/561466.html

相关文章:

  • 本地部署deepseek大模型后使用c# winform调用(可离线)
  • 图数据库Neo4j面试内容整理-使用场景-社交网络
  • 【conda环境泄露】所有环境共享了一个包?
  • Debain12.9安装NCCL GPU通讯组件
  • DeepEP通信库
  • Vue.js 编写组件单元测试
  • 一个std::async的示例
  • 【Linux进程三】进程的状态
  • XTOM工业级蓝光三维扫描仪在笔记本电脑背板模具全尺寸检测中的高效精准应用
  • 001第一个flutter文件
  • 源码分享1:批量修改PDF文件名称
  • 【MySQL篇】MySQL操作库
  • LIS系统源码,医院检验系统源码,lis软件源码
  • Java Web开发实战与项目——项目集成与部署
  • 1 什么是微服务?MybatisPlus框架
  • NVIDIA DLI引领创新课程:基于提示工程的LLM应用开发探索
  • 如何在Ubuntu 22.04或20.04 Linux上安装MobaXterm
  • 力扣2454. 下一个更大元素 IV
  • 电脑键盘知识
  • 【视频2 - 4】初识操作系统,Linux,虚拟机