解决 redis超时失效key 的监听触发时收不到消息问题

本文介绍如何使用Redis的发布订阅功能实现数据过期通知。通过修改配置文件并编写Java代码,实现了对设置过期时间的数据进行监听并在其过期时触发特定操作。

过期事件通过Redis的订阅与发布功能(pub/sub)来进行分发。

而对超时的监听呢,并不需要自己发布,只有修改配置文件redis.conf中的:notify-keyspace-events Ex,默认为notify-keyspace-events "" 

修改好配置文件后,redis会对设置了expire的数据进行监听,当数据过期时便会将其从redis中删除:

1.先写一个监听器:

package cn.uce.suc.bkg.control;
import redis.clients.jedis.JedisPubSub;
public class KeyExpiredListener extends JedisPubSub{
	public void onPSubscribe(String pattern, int subscribedChannels) {  
        System.out.println("onPSubscribe "  
                + pattern + " " + subscribedChannels);  
    }  
    @Override  
    public void onPMessage(String pattern, String channel, String message) {  
        System.out.println("onPMessage pattern "  
                        + pattern + " " + channel + " " + message);  
    }  
}

2.订阅者:

package cn.uce.suc.bkg.control;

import java.util.List;

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

public class Subscriber {
	public static void main(String[] args) {  
        JedisPool pool = new JedisPool(new JedisPoolConfig(), "127.0.0.1");  
        Jedis jedis = pool.getResource();  
        config(jedis);//最重要的部分
        jedis.psubscribe(new KeyExpiredListener(), "__keyevent@0__:expired"); 
    }  
	private static void config(Jedis jedis){
        String parameter = "notify-keyspace-events";
        List<String> notify = jedis.configGet(parameter);
        if(notify.get(1).equals("")){
            jedis.configSet(parameter, "Ex"); //过期事件
        }
    }
}

3.测试类:

package cn.uce.suc.bkg.control;

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

public class TestJedis {
	 public static void main(String[] args) {  
	        JedisPool pool = new JedisPool(new JedisPoolConfig(), "127.0.0.1");  
	        Jedis jedis = pool.getResource();  
	        //jedis.setex("notify-task-001", 10,"empty");
	        jedis.set("oneKey", "suhao");
	        jedis.expire("oneKey", 3);
	 }  
}
4.结果:

先启动订阅者,然后执行测试类,然后等待10秒之后再监听类的方法中就可以获得回调。非常需要主要的时,过期监听的管道默认是__keyevent@0__:expired,艾特后面的0表示第几个是数据库,redis默认的数据库是0~15一共16个数据库。所以如果你存入的数据库是2,那么数据接收的管道就是__keyevent@2__:expired

5.问题:

在先启动订阅者,然后执行测试类时在监听器类onPMessage方法接收不到消息,后来通过查明是因为虽然修改配置文件redis.conf中的:notify-keyspace-events Ex,但是根本没有生效,解决办法是:Subscriber类加入config私有方法判断设置notify-keyspace-events Ex是否生效,否则重新手动设置notify-keyspace-events Ex。


<think>我们正在讨论Redis的键过期事件监听配置。用户希望当Redis中的某个键过期能够触发一个事件,以便执行特定的操作。这通常用于实现缓存失效后的回调、清理资源等场景。 Redis提供了键空间通知(Keyspace Notifications)功能,允许客户端通过订阅频道来接收特定事件的通知,其中包括键过期事件。但是,这个功能默认是关闭的,需要手动配置开启。 根据引用[1],Redis内部使用事件循环(event loop)来处理I/O事件,这包括处理客户端的连接和请求。键空间通知也是基于这个事件循环机制实现的。 ### 配置步骤 1. **修改Redis配置**: 在redis.conf配置文件中,需要设置`notify-keyspace-events`参数来启用键空间通知。对于键过期事件,我们需要关注`Ex`事件(即过期事件)。可以设置为: ``` notify-keyspace-events Ex ``` 这里的`E`表示启用键空间事件(keyspace events),`x`表示启用过期事件(expired events)。 另外,也可以使用更宽松的设置,例如: ``` notify-keyspace-events A ``` 这里的`A`表示所有事件(包括键空间事件和键事件),但这样会产生大量事件,可能会影响性能,所以通常只开启需要的事件。 修改配置后需要重启Redis服务,或者通过`CONFIG SET`命令动态设置(不需要重启): ```bash CONFIG SET notify-keyspace-events Ex ``` 2. **订阅过期事件**: 客户端需要订阅特定的频道来接收过期事件。Redis会发送两种事件: - 键空间通知(keyspace notification):频道名为`__keyspace@<db>__:<key>`,事件类型为`expired`。 - 键事件通知(key-event notification):频道名为`__keyevent@<db>__:expired`,事件内容为过期的键名。 通常,我们使用键事件通知,因为我们可以直接获取到过期的键名。 例如,订阅数据库0的过期事件频道: ```bash SUBSCRIBE __keyevent@0__:expired ``` 当数据库0中有键过期,客户端会收到消息消息格式如下: ``` message __keyevent@0__:expired mykey # 这里mykey就是过期的键名 ``` ### 注意事项 1. **事件传递的可靠性**: Redis的键空间通知是一种尽力而为(best-effort)的通知机制。它并不保证事件一定会被送达,因为如果客户端在事件发生没有订阅,或者订阅后断开连接,那么就会错过事件。此外,如果Redis服务器在事件发生之后、发送通知之前崩溃,那么该事件也会丢失。 2. **事件延迟**: 过期事件是在键被删除触发的。Redis有两种删除过期键的机制: - 主动删除:当访问一个键,如果发现已过期则立即删除。 - 被动删除:由Redis后台定期随机扫描并删除过期键(默认每100毫秒一次)。 因此,即使键已经过期,事件也可能在最多100毫秒(或更多,取决于配置)后才会被触发。 3. **性能影响**: 开启键空间通知会增加Redis的CPU使用率,尤其是在有大量事件。因此,在生产环境中应谨慎使用,并确保有足够的资源。 4. **客户端实现**: 客户端需要实现一个订阅者,持续监听事件频道。在大多数编程语言中,可以使用相应的Redis客户端库来实现。 ### 示例:使用Python监听过期事件 以下是一个使用Python的redis-py库监听过期事件的简单示例: ```python import redis # 连接到Redis r = redis.Redis(host='localhost', port=6379, db=0) # 创建订阅对象 pubsub = r.pubsub() # 订阅键事件通知频道,监听过期事件 pubsub.psubscribe('__keyevent@*__:expired') # 使用模式订阅所有数据库的过期事件 # 循环处理事件 for message in pubsub.listen(): if message['type'] == 'pmessage': # 打印过期的键 print(f"Expired key: {message['data']}") ``` ### 总结 Redis的键过期事件监听通过键空间通知机制实现,需要配置`notify-keyspace-events`参数并订阅相应频道。虽然这个功能非常有用,但需要注意其可靠性和性能影响。在实际应用中,建议结合业务场景评估是否适合使用,并考虑使用其他机制(如定任务)作为补充。 如果您需要更可靠的事件传递,可能需要考虑其他方案,例如使用Redis的Streams或者消息队列。 §§相关问题§§ 1. 如何确保Redis键过期事件不丢失? 2. Redis键空间通知还有哪些其他类型的事件? 3. 在分布式系统中如何实现基于Redis过期事件的回调?</think>### Redis 键过期事件监听配置详解 要监听 Redis 键过期事件并执行特定操作,需通过 Redis 的**键空间通知(Keyspace Notifications)** 功能实现。以下是完整配置指南: #### 1. **核心机制:键空间通知** Redis 使用事件驱动架构监控键变化[^1],键过期事件属于特殊事件类型: - 键空间通知分为两类: - `__keyspace@<db>__`:事件类型通知(如 `expired`) - `__keyevent@<db>__`:键名通知(如 `:expired`) - 过期事件标识符:`Ex`(E 表示键事件,x 表示过期事件) #### 2. **配置步骤** ##### (1) 启用键空间通知 修改 `redis.conf` 配置文件: ```conf # 启用键过期事件通知 (Ex = 过期事件) notify-keyspace-events Ex # 可选:增加其他事件类型 # notify-keyspace-events AEx # A=所有事件, E=键事件, x=过期事件 ``` 或在运行动态配置: ```bash redis-cli config set notify-keyspace-events Ex ``` ##### (2) 事件订阅方式 **方式一:命令行订阅(测试用)** ```bash # 订阅数据库0的所有过期事件 redis-cli --csv psubscribe '__keyevent@0__:expired' ``` **方式二:编程实现(Python示例)** ```python import redis r = redis.Redis() pubsub = r.pubsub() pubsub.psubscribe('__keyevent@*__:expired') # 监听所有DB的过期事件 for message in pubsub.listen(): if message['type'] == 'pmessage': expired_key = message['data'].decode('utf-8') print(f"Key expired: {expired_key}") # 在此执行自定义业务逻辑 ``` #### 3. **关键注意事项** 1. **事件可靠性**: - 过期事件在键被删除触发 - Redis 定期删除过期键(默认每100ms扫描一次),事件可能有延迟 - 不保证100%送达(客户端断开会丢失事件) 2. **内存淘汰策略影响**: 当内存不足,根据配置的淘汰策略(如 `volatile-lru`)[^3],部分键可能被提前删除而不触发过期事件 3. **性能优化**: - 高频率过期场景下,建议使用单独的 Redis 实例处理事件通知 - 批量过期键处理可通过 `SCAN` 命令补偿可能丢失的事件 #### 4. **生产环境最佳实践** 1. **事件处理器设计**: ```python # 使用线程池处理事件 from concurrent.futures import ThreadPoolExecutor def handle_expiration(key): # 执行数据库清理、缓存更新等操作 pass with ThreadPoolExecutor(max_workers=10) as executor: for message in pubsub.listen(): if message['type'] == 'pmessage': executor.submit(handle_expiration, message['data']) ``` 2. **跨语言支持**: - Java:使用 Jedis 或 Lettuce 的 `RedisPubSubListener` - Node.js:使用 `ioredis` 的 `psubscribe` 方法 3. **替代方案**: - 使用 Redis Stream 作为可靠事件队列 - 通过 `EXPIRE` + `TTL` 轮询补偿(适用于低频场景) #### 5. **典型应用场景** 1. 会话管理:用户会话过期自动清理资源 2. 限流器:API调用配额到期自动重置 3. 缓存失效:缓存键过期重新加载数据 4. 分布式锁:自动释放超时锁 > ⚠️ **重要提示**:键空间通知会轻微增加CPU负载,每秒过期键超1万需进行压力测试[^1]。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值