【redis】04-redis 根据监听key的失效事件实现订单超时关闭

Redis监听key的失效事件

概念:

           当redis的key失效以后,会回调程序的监听方法。

应用场景:

1. 一个交易系统里面有一个价格提醒的功能,用户可以设置一组价格并设置一个周期,后台需要在交易的时间内进行价格扫描一旦触发用户设置的价格的周期就需要下发消息提醒给用户,提醒用户交易做单;

2. 订单超时自动关闭功能

 

现在我们就来实现订单超时自动关闭功能。

实现原理:

  1. 生成订单的时候,给redis中存入key加上过期时间,且订单状态为未支付0
  2. 过期时间到了,redis配置了key失效时间的回调,我们取出回调中的key,判断一下是否为订单的key
  3. 如果是则根据key取出订单的id,再判断订单的状态
  4. 如果订单的状态为未支付,则更新数据库,关闭订单

redis的key失效监听默认是没有开启的,需要我们手动开启。

vi redis.conf

将notify-keyspace-events "" 改为notify-keyspace-events "Ex"

然后重启redis服务即可。

SpringBoot整合redis实现订单超时关闭

RedisListenerConfig 

 @Configuration
public class RedisListenerConfig {
    @Bean
    RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
        return container;
    }
}

RedisKeyExpirationListener key失效的监听类

@Component
public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener {
    public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) {
        super(listenerContainer);
    }

    /**
     * 待支付
     */
    private static final Integer ORDER_STAYPAY = 0;
    /**
     * 失效
     */
    private static final Integer ORDER_INVALID = 2;
    @Autowired
    private OrderMapper orderMapper;

    /**
     * Redis失效事件 key
     *
     * @param message
     * @param pattern
     */
    @Override
    public void onMessage(Message message, byte[] pattern) {
        String expiraKey = message.toString();
        // 根据key查询 value 如果还还是为待支付状态 将订单改为已经超时~~
        OrderEntity orderNumber = orderMapper.getOrderNumber(expiraKey);
        System.out.println(expiraKey);
        if (orderNumber == null) {
            return;
        }
        if (orderNumber.getOrderStatus().equals(ORDER_STAYPAY)) {
            // 将订单状态改为已经失效
            orderMapper.updateOrderStatus(expiraKey, ORDER_INVALID);
        }
    }
}
  

OrderController 

@RestController
public class OrderController {
    @Autowired
    private OrderMapper orderMapper;
    @Autowired
    private RedisUtils redisUtils;

    @RequestMapping("/saveOrder")
    public String saveOrder() {
        // 1.生成token
        String orderToken = UUID.randomUUID().toString();
        String orderId = System.currentTimeMillis() + "";
        //2. 将该token存放到redis中
        redisUtils.setString(orderToken, orderId, 5L);
        OrderEntity orderEntity = new OrderEntity(null, "蚂蚁课堂永久会员", orderId, orderToken);
        int result = orderMapper.insertOrder(orderEntity);
        return result > 0 ? "success" : "fail";
    }
}

OrderEntity 

 @Data
public class OrderEntity {
    private Long id;
    private String orderName;
    /**
     * 0 待支付 1 已经支付
     */
    private Integer orderStatus;

    private String orderToken;
    private String orderId;

    public OrderEntity(Long id, String orderName, String orderId, String orderToken) {
        this.id = id;
        this.orderName = orderName;
        this.orderId = orderId;
        this.orderToken = orderToken;
    }
}
 

OrderMapper 

 public interface OrderMapper {

    @Insert("insert into order_number values (null,#{orderName},0,#{orderToken},#{orderId})")
    int insertOrder(OrderEntity OrderEntity);


    @Select("SELECT ID AS ID ,order_name AS ORDERNAME ,order_status AS orderstatus,order_token as ordertoken,order_id as  orderid FROM order_number\n" +
            "where order_token=#{orderToken};")
    OrderEntity getOrderNumber(String orderToken);

    @Update("\n" +
            "\n" +
            "update order_number set order_status=#{orderStatus} where order_token=#{orderToken};")
    int updateOrderStatus(String orderToken, Integer orderStatus);
}
 

 

 

 

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在 Redis 中,我们可以使用 KEYS 命令来获取所有的 key,但是这个命令性能比较差,因为它会遍历整个 Redis 数据库。为了避免这个问题,我们可以使用 Redis 的发布订阅机制来监听 key 失效事件。 具体步骤如下: 1. 首先我们需要在代码中订阅一个特定的频道,用于接收 key 失效事件的通知。可以使用 Redis 的 SUBSCRIBE 命令来完成订阅操作: ``` SUBSCRIBE __keyevent@0__:expired ``` 2. 接着,我们需要在 Redis设置一个 key,并为其设置过期时间。在 key 过期后,Redis自动发布一条消息到指定的频道中,我们可以在代码中接收这个消息: ``` SET mykey "hello world" EXPIRE mykey 60 ``` 3. 最后,我们需要编写一个消息处理函数,用于接收并处理 Redis 发布到频道中的消息。在 Python 中,可以使用 Redis 的 Python 客户端库 redis-py 来实现这个功能: ```python import redis r = redis.Redis(host='localhost', port=6379, db=0) def handle_message(message): print(message) pubsub = r.pubsub(ignore_subscribe_messages=True) pubsub.subscribe('__keyevent@0__:expired') for message in pubsub.listen(): handle_message(message['data']) ``` 在上面的代码中,我们首先创建了一个 Redis 客户端对象,并定义了一个消息处理函数 handle_message。然后,我们创建了一个 pubsub 对象,并使用 ignore_subscribe_messages 参数来忽略订阅时产生的消息。接着,我们使用 subscribe 方法来订阅指定的频道,并使用 listen 方法来循环接收 Redis 发布到频道中的消息,并将其传递给 handle_message 函数进行处理。 当我们运行上面的代码并设置了一个 key 的过期时间后,如果这个 key 过期了,Redis 就会自动发布一条消息到指定的频道中,我们就可以在代码中接收到这个消息,并进行相应的处理了。 ### 回答2: 监听Redis缓存过期事件Key失效事件)可以通过Redis的订阅发布功能来实现。在Redis中,可以通过配置键空间通知(keyspace notifications)来启用这个功能。 首先,我们需要在Redis配置文件中打开键空间通知功能。可以通过修改redis.conf文件中的`notify-keyspace-events`参数来开启键空间通知。例如,可以将其设置为`Ex`,表示启用键过期事件的通知。 启用键空间通知后,我们可以使用`SUBSCRIBE`命令来订阅指定的频道,以便监听特定事件。对于键过期事件,我们可以订阅`__keyevent@0__:expired`频道。 以下是监听Redis缓存过期事件的示例代码: ``` import redis def event_handler(message): print("Key expired:", message['data']) def main(): r = redis.Redis(host='localhost', port=6379, db=0) pubsub = r.pubsub() pubsub.subscribe('__keyevent@0__:expired') for message in pubsub.listen(): if message['type'] == 'message': event_handler(message) if __name__ == "__main__": main() ``` 在上述示例代码中,我们使用Python的`redis`模块来连接Redis,并创建一个订阅对象`pubsub`。然后,我们使用`subscribe`方法来订阅`__keyevent@0__:expired`频道,并通过循环监听事件。 当有键过期事件发生时,`event_handler`函数会被调用,并打印出相应的信息。 通过以上代码,我们可以在Redis实现监听缓存过期事件的功能。当监听到键过期事件时,可以做相应的处理操作,例如重新加载缓存数据或执行其他业务逻辑。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值