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

redis单线程 ,当redis在执行lua脚本的时候,会执行其他redis操作吗?

Redis有三种类型的线程:主线程,多个bio线程,后台线程。

Redis Server进程的5个线程,其中第一个是后台线程,中间三个是bio线程,最后是main主线程

Redis执行Lua脚本,已经数据操作本身,都是在主线程中完成的。同时网络相关操作是网络bio线程处理,异步的aof、过期数据清理等操作可以通过后台线程处理。

所以回到本问题,答案就是,Lua脚本执行过程中,其他的数据操作被阻塞,不会执行。

同时这里可以思考,如果不是操作数据的命令是否可以执行呢?比如ping。

我们做个实验,先在lua脚本里写点代码,让redis很长一段时间停不下来。这个期间来做其他操作。由于lua里没有sleep方法,我们用一个复杂的计算来模拟延时,比如阶乘。

用Lua写一个阶乘函数

计算一个非常大的数的阶乘,这样一时半会redis lua执行不完

然后去ping,发现无法执行。

这里可以看到提示脚本执行过程中不能执行,redis处于busy状态

也就是说,LUA脚本执行过程中,除了提示信息中的两个命令(SCRIPT KILL和SHUTDOWN NOSAVE)以外,其他的命令都无法执行。

所以,这个事实要求我们,一定不要在LUA脚本里写太多复杂逻辑。

我们之前做分布式缓存平台的时候,为了数据保持版本一致性,给各个redis写操作命令都加了一个LUA脚本,脚本多了以后,为了考虑各种通用场景和条件判断,加了很多函数、字符串处理、多层循环,后来就发现这个方式虽然使得我们的脚步通用性和可维护性好了不少,但是缓存性能下降了一个数量级,RT从1ms级别到10ms级别,我们预期是3ms级别可以接受。

为什么会这样,因为一次redis内存数据读写就几个指令,如果加上多层循环和字符串操作,这样就会在这个脚本过程执行了上百个指令,注意redis数据操作是单线程的,自然就卡住了其他并发访问的性能。合理的做法就是要让LUA脚本里的操作指令尽量简单。把这些预处理的东西,都放到Java程序里去做,充分利用JVM里的多线程和多机器节点的计算能力。


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

相关文章:

  • 【CSS】什么是BFC?
  • 写给初学者的React Native 全栈开发实战班
  • 如何向函数模块 FM 中传递 Range 参数
  • 【OpenEuler】配置虚拟ip
  • (六)Spark大数据开发实战:豆瓣电影数据处理与分析(scala版)
  • 基于Python的网上银行综合管理系统
  • 解决Springboot项目Maven下载依赖速度慢的问题
  • EmguCV学习笔记 C# 9.3 移动检测类
  • Vue(十二) Vuex、四个map方法的使用、Vuex模块化+namespace命名空间
  • Unity(2022.3.41LTS) - UI详细介绍-InputField(输入字段)
  • 使用golang的AST编写定制化lint
  • 小程序的页面跳转方式
  • C语言遇见的一些小问题
  • 使用实例:xxl-job应用在spring cloud微服务下
  • MySQL——事务与存储过程(三)存储过程的使用(1)调用存储过程
  • Unity(2022.3.41LTS) - 3D动画
  • 在 Docker 中配置 npm 和 pnpm 使用镜像源
  • np.ndarray和np.array区别;MXNet的 mx.array 类型是什么;NDArray优化了什么:并行计算优化
  • uniapp钱包支付、与设置密码页面
  • 读书笔记:《深入理解Java虚拟机》(9)
  • Baumer工业相机堡盟工业相机如何通过NEOAPI SDK使用UserSet功能保存和载入相机的各类参数(Python)
  • java导出Excel接口
  • 32位CPU中,实现32位无符号乘法,返回64位无符号数据。原理解析。
  • oracle expdp/impdp 迁移数据库
  • 数学基础 -- 图像处理之Sobel卷积核推导过程
  • C++STL~~list