在redis中调用过期或pexpire时,时间是否舍入?

问题描述 投票:1回答:1

我正在呼叫expire以获取现有的Redis密钥。假设我的值是5。如果密钥已经存在并且距离到期还有4.75秒,那么它是停留在4.75秒还是将其舍入到5秒?

我可以使用pexpire来获得更多的粒度,但是仍然存在部分毫秒的舍入问题-除非毫秒是redis中最小的粒度...

[如果有帮助,这是我的速率限制脚本,该脚本需要一个密钥,一个要增加的数量和一个毫秒级的速率限制窗口,该窗口会一直递减直到密钥掉出,此时下一个调用将添加该密钥并设置一个新鲜的到期时间。然后返回新的递增值。

local f,k,a,b,c c=ARGV[2] f=redis.call k=KEYS[1] a=f('incrby',k,ARGV[1]) b=f('pttl',k) f('pexpire',k,math.min(b<0 and c or b,c)) return a

UPDATE新的速率限制脚本没有部分时间问题,仅当密钥根本没有设置过期时,才设置过期:

local f,k,a,b f=redis.call k=KEYS[1] a=f('incrby',k,ARGV[1]) b=f('pttl',k) if b<0 then f('pexpire',k,ARGV[2]) end return a
redis lua
1个回答
0
投票

它停留在4.75秒还是四舍五入到5秒?

返回到完整的5秒钟TTL。

除非毫秒是重新分配中的最小粒度...

毫秒,对于2.6或更高版本

请参见Expire accuracy

在Redis 2.4中,过期可能不精确,并且可以在零到一秒之间。从Redis 2.6开始,过期错误是从0到1毫秒。

密钥过期信息存储为Unix绝对时间戳(在Redis 2.6或更高版本中,以毫秒为单位)。

如果要验证,可以使用一些Lua脚本

EVAL "local result = {'Time at start', 0, 'Expires in (ms)', 0, 'Time at end', 0} \n result[2] = redis.call('TIME') \n redis.call('EXPIRE', KEYS[1], ARGV[1]) \n result[4] = redis.call('PTTL', KEYS[1]) \n result[6] = redis.call('TIME') \n return result" 1 myKey 5

脚本的友好视图:

local result = {'Time at start', 0, 'Expires in (ms)', 0, 'Time at end', 0}
result[2] = redis.call('TIME') 
redis.call('EXPIRE', KEYS[1], ARGV[1]) 
result[4] = redis.call('PTTL', KEYS[1]) 
result[6] = redis.call('TIME')
return result
© www.soinside.com 2019 - 2024. All rights reserved.