说明redis.call()和redis.pcall()之间的区别。

我尝试执行下面错误的 eval 命令,以了解 redis.call() 和 redis.pcall() 之间的区别:

eval "return redis.call(ARGV[2],KEYS[1])" 1 key get
eval "return redis.pcall(ARGV[2],KEYS[1])" 1 key get

在两种情况下,我都得到了以下错误:

(错误)Lua redis() 命令参数必须是字符串或整数

这个错误并没有传达出根据文档所述redis.call()和redis.pcall()之间的区别:

“redis.call()与redis.pcall()类似,唯一的区别是,如果一个Redis命令调用将导致错误,redis.call()将引发Lua错误,进而强制EVAL返回给命令调用者一个错误,而redis.pcall将捕获错误返回代表错误的Lua表。”

因此,根据文档,在使用redis.pcall()的情况下,错误应该被捕获,对吗!在这种情况下,为什么两个错误都相同?如果我误解了差异,有人能清楚地说明这些命令之间的差异会更好!

点赞
用户1114486
用户1114486

它会失败,因为 Redis 无法执行 call 或 pcall 命令。我的意思是在执行实际的 Redis 命令之前失败了(在这里,是一个 get 命令)。pcall只会捕获 Redis 命令的执行错误,而不是 pcall 本身的执行错误。

让我们修改您的输入,以使 Redis 命令失败(而不是 redis.call 命令本身)。

> EVAL "return redis.call(ARGV[1],KEYS[1])" 1 key get
"100"

> EVAL "return redis.call(ARGV[1],KEYS[1])" 1 key born_to_fail
(error) ERR Error running script (call to f_2673dc91ae540aa65dedd262a952d5338e330b37): @user_script:1: @user_script: 1: Unknown Redis command called from Lua script

> EVAL "return redis.pcall(ARGV[1],KEYS[1])" 1 key born_to_fail
(error) @user_script: 1: Unknown Redis command called from Lua script

您可以看到,在第二个调用中,Redis 错误在 Lua 错误中显示。

在第三个调用中,使用了 pcall,所以结果不再是 Lua 错误,而是一个简单的输出字符串,其中包含错误的文本。

2013-12-18 15:09:17
用户204011
用户204011

这是一个棘手的情况,因为在你的示例中,该命令没有生成错误,你使用 redis.callredis.pcall 不正确(因为 ARGV[2]nil,就像错误信息告诉你的那样)。因此,在两种情况下,错误都无法恢复。

这里是一个实际上失败了的 命令 的示例,你可以看到区别:

redis 127.0.0.1:6379> set foo bar
OK
redis 127.0.0.1:6379> eval 'redis.call("hget","foo","bar")' 0
(error) ERR Error running script (call to f_9e6d82f0740926e0a70775430bda59a54d4e0664): ERR Operation against a key holding the wrong kind of value
redis 127.0.0.1:6379> eval 'redis.pcall("hget","foo","bar")' 0
(nil)

然而,你可能会注意到我没有返回 pcall 的结果,所以脚本返回 nil。如果我返回一个出错命令的结果会怎样呢?

redis 127.0.0.1:6379> eval 'return redis.call("hget","foo","bar")' 0
(error) ERR Error running script (call to f_d0a8dce7264708876edf262052788fc90a8e8325): ERR Operation against a key holding the wrong kind of value
redis 127.0.0.1:6379> eval 'return redis.pcall("hget","foo","bar")' 0
(error) ERR Operation against a key holding the wrong kind of value

对于 call 而言,什么都没有改变,因为错误(像其他语言中的异常 - Java、Python等)在函数返回之前就被抛出了。

但是对于 pcall 而言,函数调用返回一个带有单个 err 字段的表,这个表被 Redis 转换为 "Error reply",所以你看不到它。你该如何检查呢?把它线性化!

redis 127.0.0.1:6379> eval 'local t = redis.pcall("hget","foo","bar"); local r = {type(t)}; for k,v in pairs(t) do r[#r+1] = k; r[#r+1] = v; end; return r' 0
1) "table"
2) "err"
3) "ERR Operation against a key holding the wrong kind of value"
2013-12-18 15:10:08