当在 Redis 中使用 'redis.call' 时,如果 key 不存在,它会返回什么值?

我在这里找到了如何在 Lua 中测试变量是否为 nilLearn X in Y minutes== 表示相等,~= 表示不相等,并且 ref here。但是为什么我的代码不像那样工作呢?

在 Docker 中启动 Redis 服务器:

PS C:\Program Files\Docker\Docker\resources\bin> .\docker.exe run --name redis-work -p 2019:6379 redis

Redis 客户端在 Docker 中:

PS C:\Program Files\Docker\Docker\resources\bin> docker run -it --name redis-cli-work --link redis-work:redis --rm redis redis-cli -h redis -p 6379

第一步:

redis:6379> eval 'local r = redis.call("get", "key_which_does_not_exist") return r' 0
(nil)

第二步:

redis:6379> eval 'local r = redis.call("get", "key_which_does_not_exist") if r == nil then return \'nil for key\' else return r end' 0
(nil)

第三步:

redis:6379> eval 'local r = redis.call("get", "key_which_does_not_exist") if r ~= nil then return r else return \'nil for key\' end' 0
(nil)

如果 key 不存在,我认为 redis.call 将从第一段代码中返回 nil,但从第二段代码中返回 r == nil 却是 false,否则将打印 'nil for key',这是否意味着 r 不为 nil ?那么从第三个之中,因为r不为空,为什么它却打印 (nil),我认为只有 nil 才会打印 (nil)

所以这个值,在客户端看起来像是 nil,但在服务器端,它既不是 nil 也不是非 nil 的值,这是什么值?


问题是,如果 key_which_does_not_exist 不存在,那么 r 就是 false,如果 lua 只是将此 false 返回给 Redis,那么 Redis 客户端将变为 (nil);对于第二段代码,如果我们只想在 r == nil 为 false 时返回 r,那么 r 为 false 会导致 r == nil 的值为 false,我们返回 r 会导致返回 false,Redis 再次将 false 转换为 (nil);对于最后一段代码,r 为 false 使 r ~= nil 为 true,然后 Lua return r,实际上将 false 返回给 Redis,Redis 将 false 转换为 (nil)

问题是,在 lua 中,false 不是 nil,在 redis 中,来自 lua 的 false 将被转换为 (nil)

点赞
用户3160475
用户3160475

根据["Lua 和 Redis 数据类型转换"](https://redis.io/commands/eval#conversion-between-lua-and-redis-data-types)中的文档:

Redis 到 Lua 的转换表格

...

\* Redis 的 Nil bulk 回复和 Nil multi bulk 回复 -> Lua false 布尔类型。

所以,当 GET 返回 Redis 的 Nil 因为键不存在时,在此情况下它被表示为 Lua 的 false。

127.0.0.1:6379> eval "local r = redis.call('get','foo') if r == false then return 'falsehood' else return r end" 0
"falsehood"
2019-11-12 18:20:20