本文摘自PHP中文网,作者silencement,侵删。
先来看一个问题:
如何处理订单过期自动取消,比如下单30分钟未支付自动更改订单状态?
解决方案:
可以利用redis天然的key自动过期机制,下单时将订单id写入redis,过期时间30分钟,30分钟后检查订单状态,如果未支付,则进行处理但是key过期了redis有通知吗?答案是肯定的。
开启redis key过期提醒
修改redis相关事件配置。找到redis配置文件redis.conf,查看“notify-keyspace-events”的配置项,如果没有,添加“notify-keyspace-events Ex”,如果有值,添加Ex,相关参数说明如下:
1
2
3
4
5
6
7
8
9
10
11
K:keyspace事件,事件以__keyspace@<db>__为前缀进行发布;
E:keyevent事件,事件以__keyevent@<db>__为前缀进行发布;
g:一般性的,非特定类型的命令,比如del,expire,rename等;
$:字符串特定命令;
l:列表特定命令;
s:集合特定命令;
h:哈希特定命令;
z:有序集合特定命令;
x:过期事件,当某个键过期并删除时会产生该事件;
e:驱逐事件,当某个键因maxmemore策略而被删除时,产生该事件;
A:g
$lshzxe
的别名,因此”AKE”意味着所有事件。
redis测试:
打开一个redis-cli ,监控db0的key过期事件
1
2
3
4
5
127.0.0.1:6379> PSUBSCRIBE __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1)
"psubscribe"
2)
"__keyevent@0__:expired"
3) (integer) 1
打开另一个redis-cli ,发送定时过期key
1
127.0.0.1:6379> setex test_key 3 test_value
观察上一个redis-cli ,会发现收到了过期的keytest_key,但是无法收到过期的value test_value
1
2
3
4
5
6
7
8
9
127.0.0.1:6379> PSUBSCRIBE __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1)
"psubscribe"
2)
"__keyevent@0__:expired"
3) (integer) 1
1)
"pmessage"
2)
"__keyevent@0__:expired"
3)
"__keyevent@0__:expired"
4)
"test_key"
在springboot中使用
pom 中添加依赖
1
2
3
4
5
<!-- redis -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
定义配置RedisListenerConfig
1
2
3
4
5
6
7
8
9
10
11
import edu.zut.ding.listener.RedisExpiredListener;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.data.redis.connection.RedisConnectionFactory;import org.springframework.data.redis.listener.PatternTopic;import org.springframework.data.redis.listener.RedisMessageListenerContainer;
@Configurationpublic
class
RedisListenerConfig { @Bean
RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) {
RedisMessageListenerContainer container =
new
RedisMessageListenerContainer();
container.setConnectionFactory(connectionFactory);
return
container;
}
}
定义监听器,实现KeyExpirationEventMessageListener接口,查看源码发现,该接口监听所有db的过期事件keyevent@*:expired"
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
import edu.zut.ding.constants.SystemConstant;import edu.zut.ding.enums.OrderState;import edu.zut.ding.service.OrderService;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.data.redis.connection.Message;import org.springframework.data.redis.listener.KeyExpirationEventMessageListener;import org.springframework.data.redis.listener.RedisMessageListenerContainer;import org.springframework.stereotype.Component;
@Componentpublic
class
RedisKeyExpirationListener
extends
KeyExpirationEventMessageListener {
public
RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) { super(listenerContainer);
}
@Override
public
void onMessage(Message message, byte[] pattern) {
String expiredKey = message.toString();
if
(expiredKey.startsWith(
"Order:"
)){
}
}
}
或者打开RedisListenerConfig中 container.addMessageListener(new RedisExpiredListener(), new PatternTopic("__keyevent@0__:expired")); 注释,再定义监听器,监控__keyevent@0__:expired事件,即db0过期事件。这个地方定义的比较灵活,可以自己定义监控什么事件。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import org.springframework.data.redis.connection.Message;import org.springframework.data.redis.connection.MessageListener;
public
class
RedisExpiredListener
implements
MessageListener {
@Override
public
void onMessage(Message message, byte[] bytes) { byte[] body = message.getBody();
byte[] channel = message.getChannel();
System.out.
print
(
"onMessage >> "
);
System.out.println(String.format(
"channel: %s, body: %s, bytes: %s"
,
new
String(channel),
new
String(body),
new
String(bytes)));
}
}
以上就是redis是怎么监控失效的key 的详细内容,更多文章请关注木庄网络博客 !
相关阅读 >>
redis是怎么监控失效的key
更多相关阅读请进入《redis监控 》频道 >>
¥41.1元 机械工业出版社
本书主要讲述了数据模型、基于对象的数据库和XML、数据存储和查询、事务管理、体系结构等方面的内容。
转载请注明出处:木庄网络博客 » redis是怎么监控失效的key