【发布时间】:2016-09-17 10:08:14
【问题描述】:
最近我使用 Redis.Eval 改进了一些代码,效果很好。事实上,这太棒了,但我不明白这怎么可能。
对于 TL;DR
改进了多次使用 Redis.zcard 的 redis 代码,以使用 Redis.eval 一次。代码快了 100 倍以上(在测试环境中。在实际项目中,它快了 1000 倍以上)。我不知道为什么。有人可以解释一下吗?
代码的作用
它做了一个非常简单的任务。它接受一个字符串数组,这些字符串是存储在 Redis 中的 ZSET 的键,并对相应 ZSET 的大小求和,并返回一个整数值,即总和。
测试设置
为了尽可能多地消除外部变量,我设置了一个简单的测试环境,如下所示。
redis = Redis.new(host: '127.0.0.1', db: 1)
KEYS = 500.times.collect do |i| "KEY#{i}" end
KEYS.each do |key|
redis.zadd(key, 0, "DATA")
end
改进前
在我更改代码之前,它的工作方式如下所示。
sum = 0
KEYS.each do |key|
sum += redis.zcard(key)
end
然后我用下面一行代码测试了这段代码的速度。t = Time.now; sum=0; KEYS.each do |key| sum += redis.zcard(key) end; puts(Time.now - t)
结果打印出0.202seconds(202ms)
(请注意,我是根据测试环境和上面写的代码计算时间的,而不是真实环境)
改进后
在我使用 Lua 脚本和 EVAL 更改代码后,它的工作原理如下。
script = "
local sum = 0
for index, key in pairs(KEYS) do
sum = sum + redis.call('zcard', key);
end
return sum"
sum = redis.eval(script, KEYS)
然后,我还测量了执行上述代码所用的时间,使用以下一行代码。t = Time.now; redis.eval(script, KEYS); puts(Time.now - t)
这给了我0.001519seconds(1.5ms)。这比“改进前”代码快 134 倍。
混乱
让我更加困惑的是,一个redis.zcard(KEYS[0]) 大约需要0.000542seconds(0.542ms)。因此,redis.eval 代码在 redis 中求和 500 ZCARD 需要大约相同的时间来计算 ruby 中的 3 Redis.ZCARD。
当我第一次在我的项目中发现这一点时,我认为减少网络延迟和减少等待队列时间起到了作用。但是,当我在本地 redis 上对此进行测试时,我怀疑我的理论。完全没有网络延迟,也没有其他任务在使用 Redis。
我的理论是
- Ruby summation(
sum += redis.zcard(key)) 花费了大部分时间。 - 即使我使用的是本地主机,redis 和 ruby 通信之间也存在某种延迟。
- 在处理许多查询时,redis 存在内部延迟。 (虽然可能性不大)
谁能向我解释一下为什么这个 Redis.eval 代码非常快?谢谢!
【问题讨论】:
-
会不会是你的 lua 脚本中的时间提前返回了,而 redis.eval 语句仍在运行?
-
@mahatmanich 我相信情况并非如此。两者在测试环境和实际项目中返回完全相同的结果。
标签: ruby performance lua redis nosql