在 Redis 中调用 expire 或 pexpire 方法时,时间会被向上取整吗?

我正在调用 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

更新 新的速率限制脚本不再有部分时间问题,它只会在键没有设置过期时间时设置过期时间:

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
点赞
用户7017466
用户7017466

它会保持在4.75秒还是会被四舍五入到5秒?

它会被调整回到完整的5秒TTL。

除非毫秒是redis中最小的时间粒度…

毫秒是最小的时间粒度,适用于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
2020-01-01 03:16:03