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

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

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

过期事件通过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。


### 使用 Redis 实现超时告警机制 #### 配置 Redis 服务器 为了使 Redis 能够支持超时告警的功能,首先需要确保 Redis 正确加载配置文件。当启动 Redis ,需指定配置文件路径作为 `redis-server` 命令的第一个参数[^1]。 ```bash $ redis-server /path/to/redis.conf ``` #### 设置键的过期间 利用 Redis 提供的设置键值对及其生存间(TTL, Time To Live)的能力来触发超时事件。可以使用如下命令为某个特定的任务设定一个合理的存活期限: ```bash SET task_key "task_value" EX seconds_to_live ``` 其中 `seconds_to_live` 表示该任务允许的最大处理秒数。如果在此间内未完成,则认为发生了超时情况。 #### 订阅频道监听过期事件 Redis 支持发布订阅模式以及针对键空间的通知功能。开启此选项后,每当有键因为达到其 TTL 或者被显式删除而消失,都会向预先定义好的通道发送一条消息。这使得应用程序能够及获知哪些项已经超过了预定的间范围从而采取相应措施。 要在配置文件中启用这些通知,修改或添加以下几行到 `/etc/redis/redis.conf` 文件里: ```conf notify-keyspace-events Ex ``` 重启 Redis 后生效。此客户端就可以连接上并监视特定前缀下的所有对象变化状况了。 下面是一个简单的 Python 客户端例子用于接收来自 Redis 的过期通知,并据此发出警告信号: ```python import redis def listen_for_expired_keys(): r = redis.StrictRedis(host='localhost', port=6379, db=0) pubsub = r.pubsub() pubsub.psubscribe('__keyevent@0__:expired') while True: message = pubsub.get_message() if message and isinstance(message['data'], bytes): expired_key = message['data'].decode('utf-8') print(f'Warning: Key {expired_key} has been expired.') listen_for_expired_keys() ``` 这段脚本会持续运行直到手动终止它;期间任何匹配规则的对象一旦失效就会立即打印出相应的提示信息给管理员查看。 #### 结合实际场景应用 考虑到某些情况下可能不仅仅只是简单地显示文字提醒这么容易解决问题——也许还需要进一步的动作比如邮件报警、短信推送或者其他形式的通知方式。这可以根据业务需求扩展上述基础框架,在检测到异常之后调用第三方 API 接口或者执行其他自定义操作流程。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值