我正在呼叫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
它停留在4.75秒还是四舍五入到5秒?
返回到完整的5秒钟TTL。
除非毫秒是重新分配中的最小粒度...
毫秒,对于2.6或更高版本
在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