Lua脚本用于Redis,用于对键的值进行求和。

我正在构建我的第一个 Redis 服务器端脚本(用于调试),我的 Lua 经验不足使我陷入困境。

基本上,我有一个 K/V 对的数据集(包含约 1000 个值),我想列出与模式匹配的所有 KEY。例如,在 Redis-cli 中:

> KEYS "carlos:*"
1) "carlos:1"
2) "carlos:2"
3) "carlos:3"
4) "carlos:4"

基于上述输出,我想通过执行 Lua 脚本返回这些键的总和。当前,我在 sum.lua 上有以下内容:

local sum = 0
local matches = redis.call('KEYS', 'carlos:*')

for unpack(matches)
   sum = sum + redis.call('GET', matches)
end

return sum

虽然上面的脚本可能不正确,但是仅执行 redis.call('KEYS', 'carlos:*') 即产生以下错误

root@carlos:~# redis-cli EVAL "$(cat sum.lua)"

(error) ERR wrong number of arguments for 'eval' command

我尝试了许多语法迭代,但都没有成功。有什么想法吗?

谢谢

点赞
用户142162
用户142162
  1. EVAL 命令需要最少两个参数:脚本和传递给脚本的键的数量。在这种情况下,您没有传递任何键,意味着可以按照以下方式调用脚本:
redis-cli EVAL "$(cat sum.lua)" 0

或者:

redis-cli --eval sum.lua
  1. 您迭代KEYS返回的值的循环结构不正确;我已经为您修复了它。

  2. 您需要使用Lua的tonumber函数将从GET返回的值从字符串转换为数字。

通过上述更改,以下脚本应该适合您:

local sum = 0
local matches = redis.call('KEYS', 'carlos:*')

for _, key in ipairs(matches) do
    local val = tonumber(redis.call('GET', key))
    sum = sum + val
end

return sum
2014-02-11 16:25:03
用户6728052
用户6728052

对于初学者的一些重要要点:

KEYS和ARGV:这些是在Redis中Lua脚本中接收数据的Lua表。

  • 表是关联数组,并且是Lua结构化数据的唯一机制。您可以将它们视为您最熟悉的语言中的数组等效物。表是基于1的索引,即索引从1开始。因此,mytable中的第一个元素是mytable[1],第二个元素是mytable[2]等。

  • 表格不能包含空值。如果操作将生成一个表[1,nil,3,4],则结果将为[1] - 表被截断到第一个nil值。

注意:**在为Redis编写Lua脚本时,应仅使用KEYS表访问访问的每个键。 ARGV表用于参数传递**

我认为您现在可以将键和参数从任何语言传输到Lua脚本中了。

现在,我们可以继续使用Lua的[redis call](https://redis.io/commands/eval)调用redis命令[KEYS](https://redis.io/commands/keys)。

要迭代返回的数据,您可以使用lua的pairsipairs选项,并且可以在[此处](https://stackoverflow.com/questions/55108794/what-is-the-difference-of-pairs-vs-ipairs-in-lua)找到一些快速的区别。对于上述情况,ipairs很好。

现在,我们可以继续处理数据类型问题

Lua和Redis具有不同的类型系统,因此了解通过Redis-Lua边界时值的变化方式非常重要。当数字从Lua返回Redis客户端时,它变成了整数 - 小数点后的任何数字都被丢弃:

local indiana_pi = 3.2
return indiana_pi

当您运行此脚本时,Redis将返回一个整数3 - 您将失去pi的有趣部分。看起来很简单,但是当您开始在脚本中使用Redis时,情况会变得有些棘手。例如:

local indiana_pi = 3.2
redis.call("SET", "pi", indiana_pi)
return redis.call("GET", "pi")

这里的结果值是一个字符串:“3.2”。为什么?Redis没有专用的数字类型。当我们首次设置值时,Redis将其保存为字符串,丢失Lua最初将该值视为float的所有记录。当我们稍后提取该值时,它仍然是字符串。

使用GET / SET访问的Redis中的值除在运行像INCRDECR的数字操作时应将其视为字符串。这些特殊的数字操作实际上将返回整数回复,并根据数学规则操作存储的值,但Redis中存储的值的“类型”仍然是字符串值。

上面的答案摘自这篇非常棒的博客文章:

[https://www.redisgreen.net/blog/intro-to-lua-for-redis-programmers/](https://www.redisgreen.net/blog/intro-to-lua-for-redis-programmers/)

2020-05-29 03:07:20