首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >lua执行redis脚本找不到脚本的问题

lua执行redis脚本找不到脚本的问题

作者头像
用户7634691
发布于 2020-09-03 08:37:48
发布于 2020-09-03 08:37:48
3.2K00
代码可运行
举报
运行总次数:0
代码可运行

写在前面

最近遇到了一个坑,给大家分享下。

有个项目,利用redis做统计功能。一向对性能追求极致的我怎么能随便写几条redis的统计语句就应付呢。于是我打算使用lua脚本把用到的几条redis指令封装一起,这样减少和redis的IO交互,还可以保证操作原子性。我为自己的聪明才智沾沾自喜。

脚本如下(下面并不是我项目中实际的脚本,做了一些修改,大家不用纠结语法和能否运行。不过不影响本文的分析):

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private final static String luaScript = 
            "redis.call('ZREMRANGEBYSCORE',KEYS[1],0,ARGV[1]);" +
            "redis.call('ZADD', KEYS[1], ARGV[3], ARGV[4]);" +
            "redis.call('EXPIRE',KEYS[1],ARGV[2]);" +
            "end;";

然后我的java应用层的代码是这样写的:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private String luaSha;

private void runSha(String key, String expire, String score, string value) {
        if (luaSha == null) {
            luaSha = redisService.scriptLoad(luaScript, key);
        }
        redisService.evalsha(luaSha, 1, new String[]{key, expire, score, value});
    }

上面的代码我本地自测没有问题。于是自信满满的转给了测试小姐姐,我就开心的摸鱼去了。

问题来了

就在我专心致志的摸鱼的时候,测试小姐姐突然反馈,统计的结果和实际不符合,并且服务器上有一些错误日志。

日志如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
error:redis.clients.jedis.exceptions.JedisNoScriptException: NOSCRIPT No matching script. Please use EVAL
...

我看到日志的第一反应是,一定是redis配置问题,我本地测试过明明没有问题的。本着负责任的态度我还是去网上查了下这个报错。一查之后尴尬了,发现还真是自己考虑不周全。

要理解这个问题,先引出一个概念,就是redis集群里slot的概念。

使用redis-cluster集群部署Redis,redis-cluster把所有的物理节点映射到[0-16383]slot上。

比如,现在有3台Redis节点 ,分别给他们分配slot :

节点

集群slot

A

0~5000

B

5001~10000

C

10000~16383

有一个key要set到redis,先对key做hash计算然后mod 163838,比如结果是1000,那么这个key就会保存在A节点。读的时候也是一样的原理。

lua脚本有一种缓存机制。在redis集群中,为了避免重复发送脚本数据浪费网络资源,可以使用script load命令进行脚本数据缓存,并且返回一个哈希码作为脚本的调用句柄,每次调用脚本只需要发送哈希码来调用即可。

而这个脚本缓存有点像本地内存一样,需要每个节点都有缓存才可以,否则就会报上面的那个错误。那么节点上的缓存是什么加载的呢?就是下面这行代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
luaSha = redisService.scriptLoad(luaScript, key);

redis会首先根据key找到对应的slot,然后根据slot加载到对应节点上。

现在问题其实已经呼之欲出了,我们前面的java代码,只要luaSha != null就会去调用redis的evalhash执行脚本,但是因为key不是固定的(实际项目中这个key是用户id),所以有可能对应的节点上是没有脚本缓存的。

解决方案

了解了出错的原因,解决方案其实就很简单了。执行evalsha方法的时候,如果触发了JedisNoScriptException这个异常,就重新scriptLoad下脚本到缓存。这里还加了scriptExist再次检查下脚本是否存在,双重保险。

优化后的代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
private void runSha(String key, String expire, String score, string value) {
        if (luaSha == null) {
            luaSha = redisService.scriptLoad(luaScript, key);
        }
        try {
            redisService.evalsha(luaSha, 1, new String[]{key, expire, score, value});
        } catch (JedisNoScriptException e) {
            boolean scriptExist = redisService.scriptExist(luaSha, key);
            if (!scriptExist) {
                luaSha = redisService.scriptLoad(luaScript, key);
            }
            redisService.evalsha(luaSha, 1, new String[]{key, expire, score, value});
        } catch (Exception e) {
            log.error("redis eval sha error:", e);
        }

        redisService.evalsha(luaSha, 1, new String[]{key, expire, score, value});
    }
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-09-01,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 犀牛的技术笔记 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
高并发秒杀系统实战(Redis+Lua分布式锁防超卖与库存扣减优化)
电商秒杀场景具有瞬时高并发、资源竞争激烈和数据一致性要求高三大特征。当数万用户同时抢购少量商品时(如1000件商品承受10万QPS),系统面临多重挑战:
大熊计算机
2025/07/15
960
高并发秒杀系统实战(Redis+Lua分布式锁防超卖与库存扣减优化)
【📕分布式锁通关指南 03】通过Lua脚本保证redis操作的原子性
官方介绍它是一种轻量小巧的脚本语言,设计的目的是为了嵌入应用程序,从而为应用程序提供灵活的扩展性和定制功能。redis支持嵌入Lua脚本,因此可以很方便地使用。安装过程很简单,本文不做赘述,直接移步官网下载即可,如遇到问题,善用一下你的搜索引擎即可。
别惹CC
2025/02/21
2711
【📕分布式锁通关指南 03】通过Lua脚本保证redis操作的原子性
干货|RedisTemplate调lua踩了个坑
RedisTemplate的Serializer将不同的数据类型进行序列化时,内置了一套逻辑,譬如Long类型的追回“L”,Byte的追加“B”。这就可能与redis期望的数据不一致,而引发错误
烟雨平生
2023/03/07
2.1K0
干货|RedisTemplate调lua踩了个坑
Redis进阶应用:Redis+Lua脚本实现复合操作
Redis是高性能的key-value数据库,在很大程度克服了memcached这类key/value存储的不足,在部分场景下,是对关系数据库的良好补充。得益于超高性能和丰富的数据结构,Redis已成为当前架构设计中的首选key-value存储系统。
宜信技术学院
2019/08/08
1.7K0
Redis进阶应用:Redis+Lua脚本实现复合操作
lua脚本-redis批量设置key并且带TTL-两种方案
KEYS: ["my:key:::1","my:key:::2","my:key:::3"]
王宝
2024/11/25
2120
深入浅出Redis(十):Redis的Lua脚本
Redis是一款基于内存的键值对数据库,提供了多种数据结构存储数据,存取数据的速度还非常快,除了这些优点它还提供了其他特色功能,比如:管道、lua脚本、发布订阅模型
菜菜的后端私房菜
2024/09/20
3590
【Redis】已解决:redis.clients.jedis.exceptions.JedisNoScriptException
在使用Redis进行缓存或数据处理时,Lua脚本是一种常用的扩展手段,用于保证操作的原子性和高效性。Jedis是Java中使用Redis的一个常用客户端库,但在执行Lua脚本时,开发者可能会遇到redis.clients.jedis.exceptions.JedisNoScriptException报错。这个异常通常发生在尝试执行一个未加载到Redis服务器中的脚本时。以下是一个典型场景:
屿小夏
2025/05/24
1350
Redis+Lua 实现消息和接口幂等性
为了防止消息重复消费导致业务处理异常,消息队列RocketMQ版的消费者在接收到消息后,有必要根据业务上的唯一Key对消息做幂等处理。本文介绍消息幂等的概念、适用场景以及处理方法。
Tinywan
2023/09/18
9380
Redis+Lua 实现消息和接口幂等性
redis入门(三)
在前两章介绍了Redis的一些常用的API与功能,在本章会对一些其他功能包括事务、脚本、Redis集群搭建工具以及集群动态扩容与故障转移方式进行讲解。
用户6786055
2019/12/11
8470
在Redis中使用简单强大的Lua脚本
前段时间写Redis分布式锁,想着在小灰文章的基础上再总结一下,这样能有更深的印象,顺便把Lua脚本分享一下,如果项目中使用Redis比较多,那么Lua脚本一定是会用到的,因为它简单强大。
Java识堂
2019/11/04
2.5K0
Redis常用技术-----使用Lua语言
Redis命令的计算能力并不算很强大,使用Lua语言则可以在很大程度上弥补Redis的这个不足。在Redis中,执行Lua语言是原子性,也就是说Redis执行Lua的时候是不会被中断的,具备原子性,这个特性有助于Redis对并发数据一致性的支持。
秃头哥编程
2019/06/17
8850
Redis常用技术-----使用Lua语言
玩转 lua in Redis
一、引言 Redis学了一段时间了,基本的东西都没问题了。从今天开始讲写一些redis和lua脚本的相关的东西,lua这个脚本是一个好东西,可以运行在任何平台上,也可以嵌入到大多数语言当中,来扩展其功能。lua脚本是用C语言写的,体积很小,运行速度很快,并且每次的执行都是作为一个原子事务来执行的,我们可以在其中做很多的事情。由于篇幅很多,一次无法概述全部,这个系列可能要通过多篇文章的形式来写,好了,今天我们进入正题吧。 二、Lua简介 Lua 是一个小巧的脚本语言。是巴西里约热内卢天主教大学(Pontifical Catholic University of Rio de Janeiro)里的一个研究小组,由Roberto Ierusalimschy、Waldemar Celes 和 Luiz Henrique de Figueiredo所组成并于1993年开发。 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。Lua由标准C编写而成,几乎在所有操作系统和平台上都可以编译,运行。Lua并没有提供强大的库,这是由它的定位决定的。所以Lua不适合作为开发独立应用程序的语言。Lua 有一个同时进行的JIT项目,提供在特定平台上的即时编译功能。 Lua脚本可以很容易的被C/C++ 代码调用,也可以反过来调用C/C++的函数,这使得Lua在应用程序中可以被广泛应用。不仅仅作为扩展脚本,也可以作为普通的配置文件,代替XML,ini等文件格式,并且更容易理解和维护。 Lua由标准C编写而成,代码简洁优美,几乎在所有操作系统和平台上都可以编译,运行。一个完整的Lua解释器不过200k,在目前所有脚本引擎中,Lua的速度是最快的。这一切都决定了Lua是作为嵌入式脚本的最佳选择。 三、使用Lua脚本的好处 1、减少网络开销:可以将多个请求通过脚本的形式一次发送,减少网络时延和请求次数。 2、原子性的操作:Redis会将整个脚本作为一个整体执行,中间不会被其他命令插入。因此在编写脚本的过程中无需担心会出现竞态条件,无需使用事务。 3、代码复用:客户端发送的脚步会永久存在redis中,这样,其他客户端可以复用这一脚本来完成相同的逻辑。 4、速度快:见 与其它语言的性能比较, 还有一个 JIT编译器可以显著地提高多数任务的性能; 对于那些仍然对性能不满意的人, 可以把关键部分使用C实现, 然后与其集成, 这样还可以享受其它方面的好处。 5、可以移植:只要是有ANSI C 编译器的平台都可以编译,你可以看到它可以在几乎所有的平台上运行:从 Windows 到Linux,同样Mac平台也没问题, 再到移动平台、游戏主机,甚至浏览器也可以完美使用 (翻译成JavaScript). 6、源码小巧:20000行C代码,可以编译进182K的可执行文件,加载快,运行快。 四、redis和lua整合详解 1、调用Lua脚本的语法: $ redis-cli --eval path/to/redis.lua KEYS[1] KEYS[2] , ARGV[1] ARGV[2] ... --eval,告诉redis-cli读取并运行后面的lua脚本 path/to/redis.lua,是lua脚本的位置 KEYS[1] KEYS[2],是要操作的键,可以指定多个,在lua脚本中通过KEYS[1], KEYS[2]获取 ARGV[1] ARGV[2],参数,在lua脚本中通过ARGV[1], ARGV[2]获取。 注意: KEYS和ARGV中间的 ',' 两边的空格,不能省略。
sunsky
2020/08/20
6760
Redis 如何实现延时任务队列
顾名思义,延迟队列就是进入该队列的消息会被延迟消费的队列。而一般的队列,消息一旦入队了之后就会被消费者马上消费。
Tinywan
2024/05/11
8470
Redis 如何实现延时任务队列
jediscluster.set加锁_redislock
本人已实践的参考:https://blog.csdn.net/NullToSay/article/details/109813194
全栈程序员站长
2022/11/04
5760
基于Redis和Lua的分布式限流
 Java单机限流可以使用AtomicInteger,RateLimiter或Semaphore来实现,但是上述方案都不支持集群限流。集群限流的应用场景有两个,一个是网关,常用的方案有Nginx限流和Spring Cloud Gateway,另一个场景是与外部或者下游服务接口的交互,因为接口限制必须进行限流。
程序员历小冰
2019/04/07
1.9K0
基于Redis和Lua的分布式限流
Redis入坟(二)高级特性,发布订阅、事务、Lua脚本
前面我们说通过队列的 rpush 和 lpop 可以实现消息队列(队尾进队头出),但是消费者需要不停地调用 lpop 查看 List 中是否有等待处理的消息(比如写一个 while 循环)。
源码之路
2020/09/04
9710
Redis入坟(二)高级特性,发布订阅、事务、Lua脚本
redis分布式锁的实现(setNx命令和Lua脚本)
本篇文章主要介绍基于Redis的分布式锁实现到底是怎么一回事,其中参考了许多大佬写的文章,算是对分布式锁做一个总结
全栈程序员站长
2022/08/31
2.9K0
Redis Lua脚本小学教程
Redis提供了丰富的指令集,但是仍然不能满足所有场景,在一些特定场景下,需要自定义一些指定来完成某些功能。因此,Redis提供了Lua脚本支持,用户可以自己编写脚本来实现想要的功能。
Jackeyzhe
2020/03/11
1.2K0
初学乍练redis:事务与脚本
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/wzy0623/article/details/82350861
用户1148526
2019/05/25
1.1K0
高性能伪事务之Lua in Redis
Redis2.6加入了对Lua脚本的支持。Lua脚本可以被用来扩展Redis的功能,并提供更好的性能。
sunsky
2020/08/20
2.4K0
相关推荐
高并发秒杀系统实战(Redis+Lua分布式锁防超卖与库存扣减优化)
更多 >
交个朋友
加入[架构及运维] 腾讯云技术交流站
云架构设计 云运维最佳实践
加入架构与运维趋势交流群
技术趋势前瞻 架构演进方向
加入架构与运维学习入门群
系统架构设计入门 运维体系构建指南
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验