我们有以下用例:每当某个密钥过期时,我们都需要根据其值来得到通知并执行某些操作。但是,当redis触发该expired事件时,当我们稍后尝试访问它时,该密钥已经从数据库中删除了,这当然是可以预期的。
expired
现在,有一种方法可以在过期后再次访问该条目?我猜不会。
第二种选择:有没有一种方法可以让redis在发送这些事件时发布整个值对象而不是仅仅发布键?我想可以通过Lua添加它,但是如果可能的话,我希望有一个更简单的选择。对于其他事件,我们也需要这种行为,基本上,我们需要所有通知来发布值,而不是键(我们可以GET在事件收到后立即执行一次,但是我们想绕开第二个调用,主要是拥有一个原子过程,因为值可能会在发布事件和执行GET检索值之间更改。
GET
希望这是可以理解的。也许我们看不到明显的东西,所以先谢谢!
Eli链接到的功能使您可以在密钥过期时进行监听。但是,它不能为您提供密钥的值。此外,基于已归档的github问题,您似乎无法期望很快就可以构建此功能(https://github.com/antirez/redis/issues/1876)。我使用的解决方案是创建一个特殊的“影子”过期密钥,该密钥链接到您具有实际值的密钥。
因此,假设您有一个名为的密钥testkey,它的整数值为100。此外,密钥将在10秒后过期,此时您要获取密钥的值。(也许您在存在的10秒钟内递增了密钥)。
testkey
100
首先,您需要设置侦听键空间事件。特别是您想听expired事件。您可以从配置中执行此操作,也可以config set在redis中使用命令。(有关更多信息,请参见此处:http : //redis.io/topics/notifications)
config set
CONFIG SET notify-keyspace-events Ex
现在,您可以订阅一个特殊的keyevent频道,在该频道中,您将收到有关密钥已过期的通知。
keyevent
SUBSCRIBE __keyevent@0__:expired
要订阅的频道格式为__keyevent@<db>__:<eventName>。在我们的示例中,我们假设我们正在使用默认数据库0,并且我们想监听expired事件。
__keyevent@<db>__:<eventName>
当testkey到期时,你现在会在收到一条消息,__keyevent__该消息是过期键的名称通道。当然,此时密钥已消失,因此我们无法再访问该值!解决方案是使用特殊的到期密钥。
__keyevent__
创建时,testkey还创建一个特殊的“影子”密钥(不要使实际的密钥失效testkey)。例如:
SET testkey 100 SET shadowkey:testkey "" EX 10
现在,在__keyevent@0__:expired频道中,您会收到一条消息,告诉您密钥shadowkey:testkey已过期。获取消息的值(这是键的名称),在冒号(或您决定使用的分隔符)上分割,然后手动获取并删除键的值。
__keyevent@0__:expired
shadowkey:testkey
// set your key value SET testkey 100 //set your "shadow" key, note the value here is irrelevant SET shadowkey:testkey "" EX 10 // Get an expiration message in the channel __keyevent@0__:expired // Split the key on ":", take the second part to get your original key // Then get the value and do whatever with it GET testkey // Then delete the key DEL testkey
请注意,没有使用shadowkey的值,因此您要使用尽可能小的值,该值为空字符串""。设置工作要多一些,但是上面的系统完全可以满足您的需求。开销是实际需要检索和删除密钥的一些额外命令,以及空密钥的存储成本。
""