Redis使用Lua脚本命令详解

 更新时间:2023年12月05日 11:00:54   作者:bug生产者  
这篇文章主要为大家介绍了Redis使用Lua脚本命令详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

Lua脚本

redis可以支持lua脚本,可以使用lua脚本来将几个命令整合为一个整体来执行,这样可以使得多个命令原子操作,且可以减少网络开销

Lua的数据类型

Lua是一个动态类型的语言,一个变量可以存储任何类型的值,类型有:

  • 空:nil,也就是还没有赋值
  • 字符串:用单引号 或者 双引号
  • 数字:包含整数和浮点型
  • 布尔:boolean
  • 表:表是Lua唯一的数据结构,既可以当数组,也可以做Map,或被视为对象
  • 函数:封装某个或某些功能
  • userData:用来将任意 C 数据保存在 Lua 变量中,这样的操作只能通过 C API
  • Thread:用来区别独立的执行线程,它被用来实现 coroutine (协同例程)

eval命令

redis内有内置的lua解释器,可以使用eval命令对lua脚本进行求值

# script是lua脚本
# numkeys指定键名参数的个数
# key [key ...]  在脚本中使用的redis键,个数为numkeys指定的个数,可以在lua中通过全局变量KEYS数组,从1开始,KEYS[1],KEYS[2]等
# arg [arg ...]  参数,可以在lua中通过全局变量ARGV数组访问,ARGV[1],ARGV[2]等
eval script numkeys key [key ...] arg [arg ...]
#示例:
eval "return {KEYS[1],KEYS[2],ARGV[1]}" 2 key1 key2 first

还可以在lua脚本中调用redis命令,使用redis.call

eval "return redis.call('set',KEYS[1],'bar')" 1 foo

需要注意的是,redis执行lua脚本和普通命令一样,都是会写入AOF文件和发布至主从复制连接上的,有两种方式

  • 第一种方式:和普通命令相同,涉及到的写操作都会记录/发送

    普通命令会转化为redis通信协议的格式,比起lua脚本来说,浪费了更多的带宽,而且slave接收到之后还需要再转换为普通命令

  • 第二种方式:直接将lua脚本记录/发送

    如果直接发送lua脚本,有些命令可能会导致每个机器执行的结果不同,如取随机数等

    这个redis会决定采取哪种方式来执行,在执行前会进行检测写操作是否执行了RANDOMKEY命令,来决定使用哪种方式

evalsha命令

考虑到脚本比较长的情况,如果每次调用脚本都需要将整个脚本传给redis会占用较多的带宽。为了解决该问题,redis提供了evalsha命令允许开发者通过脚本内容的SHA1摘要来执行脚本,该命令的用法的eval一样,不过是将脚本内容替换为脚本内容的SHA1摘要。

redis在执行eval命令时会计算脚本的SHA1摘要并记录在脚本缓存中,执行evalsha命令时redis会根据提供的摘要从脚本缓存中查找对应的脚本内容,如果找到了则执行脚本,否则返回错误

使用流程

  • 先计算脚本的SHA1摘要,并使用evalsha命令执行脚本
  • 获的返回值,如果返回"NoScript"错误则使用eval命令重新执行脚本

其他不常用的命令

这些命令不是不常用,而是经常被工具类封装起来,开发者如果不深入源码查看,很少会用到

将脚本加入缓存

SCRIPT LOAD命令,作用是每次执行eval命令时redis都会将脚本的SHA1摘要加入到脚本缓存中,以便下次客户端可以使用evalsha命令调用该脚本。如果只是希望将脚本加入缓存而不执行,则使用SCRIPT LOAD命令,返回值是脚本的SHA1摘要

判断脚本是否被缓存

SCRIPT EXISTS命令,可以同时查找多个脚本的SHA1摘要是否被缓存

清空脚本缓存

SCRIPT FLUSH命令,redis将脚本的SHA1摘要加入到脚本缓存后会永久保留,不会删除,可以使用该命令清空脚本缓存

强制终止当前脚本的执行

SCRIPT KILL命令,如果想终止当前正在执行的脚本可以使用该命令

以上就是Redis使用Lua脚本命令详解的详细内容,更多关于Redis使用Lua脚本的资料请关注脚本之家其它相关文章!

相关文章

  • Redis分布式缓存与秒杀

    Redis分布式缓存与秒杀

    这篇文章主要介绍了Redis分布式缓存与秒杀,单点Redis的问题,主要有数据丢失,并发能力,故障恢复,存储能力,想进一步了解的同学,可以借鉴本文
    2023-04-04
  • 使用Spring Boot实现Redis键过期回调功能示例详解

    使用Spring Boot实现Redis键过期回调功能示例详解

    这篇文章主要介绍了使用Spring Boot实现Redis键过期回调功能,就是一个实现Redis键过期回调功能的Spring Boot应用的示例,代码简单易懂,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-07-07
  • Redis中过期键删除的三种方法

    Redis中过期键删除的三种方法

    Redis中可以设置键的过期时间,并且通过取出过期字典(expires dict)中键的过期时间和当前时间比较来判断是否过期,那么一个过期的键是怎么被删除的呢?本文给大家总结了三种方法,选了其中两种给大家详细的介绍一下,需要的朋友可以参考下
    2024-05-05
  • Redis内存碎片率调优处理方式

    Redis内存碎片率调优处理方式

    Redis集群因内存碎片率超过1.5触发告警,分析发现内因与外因导致内存碎片,内因为操作系统内存分配机制,外因为Redis操作特性,使用Redis内置内存碎片清理机制可有效降低碎片率,但需注意可能影响性能,建议使用MEMORY命令诊断内存使用情况,合理配置参数以优化性能
    2024-09-09
  • Redis中实现查找某个值的范围

    Redis中实现查找某个值的范围

    这篇文章主要介绍了Redis中实现查找某个值的范围,本文的题引来了Redis作者Salvatore Sanfilippo(@antirez)的回答,比较经典,需要的朋友可以参考下
    2015-06-06
  • 浅谈Redis缓存更新策略

    浅谈Redis缓存更新策略

    这篇文章主要介绍了Redis缓存更新策略的相关资料,讲解的十分细致,有需要的小伙伴可以参考下
    2022-08-08
  • redis中RDB(Redis Data Base)的机制

    redis中RDB(Redis Data Base)的机制

    本文主要介绍了redis中RDB(Redis Data Base)的机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • websocket+redis动态订阅和动态取消订阅的实现示例

    websocket+redis动态订阅和动态取消订阅的实现示例

    本文主要介绍了websocket+redis动态订阅和动态取消订阅,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • Redis的Sentinel解决方案介绍与运行机制

    Redis的Sentinel解决方案介绍与运行机制

    这篇文章主要介绍了Redis的Sentinel解决方案介绍与运行机制, Sentinel 是一款面向分布式服务架构的轻量级流量控制组件,主要以流量为切入点,从流量控制、熔断降级、系统自适应保护等多个维度来保障服务的稳定性,需要的朋友可以参考下
    2023-07-07
  • 详解redis大幅性能提升之使用管道(PipeLine)和批量(Batch)操作

    详解redis大幅性能提升之使用管道(PipeLine)和批量(Batch)操作

    这篇文章主要介绍了详解redis大幅性能提升之使用管道(PipeLine)和批量(Batch)操作 ,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
    2016-12-12

最新评论