redis实现消息队列

用Redis实现一个消息通知系统,总结了一下技术细节,其中演示代码如果没有特殊说明,使用的都是PhpRedis扩展来实现的。
内存
比如要推送一条全局消息,如果真的给所有用户都推送一遍的话,那么会占用很大的内存,实际上不管粘性有多高的产品,活跃用户同全部用户比起来,都会小很多,所以如果只处理登录用户的话,那么至少在内存消耗上是相当划算的,至于未登录用户,可以推迟到用户下次登录时再处理,如果用户一直不登录,就一了百了了。
队列
当大量用户同时登录的时候,如果全部都即时处理,那么很容易就崩溃了,此时可以使用一个队列来保存待处理的登录用户,如此一来顶多是反应慢点,但不会崩溃。
Redis的LIST数据类型可以很自然的创建一个队列,代码如下:

<?php
class RedisClient extends Redis
{
    const POSITION_FIRST = 0;
    const POSITION_LAST = -1;
    public function zPop($zset)
    {
        return $this->zsetPop($zset, self::POSITION_FIRST);
    }
    public function zRevPop($zset)
    {
        return $this->zsetPop($zset, self::POSITION_LAST);
    }
    private function zsetPop($zset, $position)
    {
        $this->watch($zset);
        $element = $this->zRange($zset, $position, $position);
        if (!isset($element[0])) {
            return false;
        }
        if ($this->multi()->zRem($zset, $element[0])->exec()) {
            return $element[0];
        }
        return $this->zsetPop($zset, $position);
    }
}
?>
模拟实现了POP操作后,我们就可以使用ZSET实现队列了,代码如下:
<?php
$redis = new RedisClient;
$redis->connect('/tmp/redis.sock');
$redis->zAdd('msg', <PRIORITY>, <MSGID>);
while ($msg = $redis->zRevPop('msg')) {
    var_dump($msg);
}
?>

推拉
以前微博架构中推拉选择的问题已经被大家讨论过很多次了。实际上消息通知系统和微博差不多,也存在推拉选择的问题,同样答案也是类似的,那就是应该推拉结合。具体点说:在登陆用户获取消息的时候,就是一个拉消息的过程;在把消息发送给登陆用户的时候,就是一个推消息的过程。
速度
假设要推送一百万条消息的话,那么最直白的实现就是不断的插入,代码如下:

<?php
for ($msgid = 1; $msgid <= 1000000; $msgid ) {
    $redis->sAdd('usr:<USRID>:msg', $msgid);
}
?>
Redis的速度是很快的,但是借助PIPELINE,会更快,代码如下:
<?php
for ($i = 1; $i <= 100; $i ) {
    $redis->multi(Redis::PIPELINE);
    for ($j = 1; $j <= 10000; $j ) {
        $msgid = ($i - 1) * 10000 $j;
        $redis->sAdd('usr:<USRID>:msg', $msgid);
    }
    $redis->exec();
}
?>

说明:所谓PIPELINE,就是省略了无谓的折返跑,把命令打包给服务端统一处理。
前后两段代码在我的测试里,使用PIPELINE的速度大概是不使用PIPELINE的十倍。
查询
我们用Redis命令行来演示一下用户是如何查询消息的。
先插入三条消息,其分别是1,2,3:

redis> HMSET msg:1 title title1 content content1
redis> HMSET msg:2 title title2 content content2
redis> HMSET msg:3 title title3 content content3

再把这三条消息发送给某个用户,其是123:

redis> SADD usr:123:msg 1
redis> SADD usr:123:msg 2
redis> SADD usr:123:msg 3

此时如果简单查询用户有哪些消息的话,无疑只能查到一些:
redis> SMEMBERS usr:123:msg

1) "1"
2) "2"
3) "3"

如果还需要用程序根据再来一次查询无疑有点低效,好在Redis内置的SORT命令可以达到事半功倍的效果,实际上它类似于SQL中的JOIN:
redis> SORT usr:123:msg GET msg:*->title
1) "title1"
2) "title2"
3) "title3"
redis> SORT usr:123:msg GET msg:*->content
1) "content1"
2) "content2"
3) "content3"

SORT的缺点是它只能GET出字符串类型的数据,如果你想要多个数据,就要多次GET:

redis> SORT usr:123:msg GET msg:*->title GET msg:*->content
1) "title1"
2) "content1"
3) "title2"
4) "content2"
5) "title3"
6) "content3"

很多情况下这显得不够灵活,好在我们可以采用其他一些方法平衡一下利弊,比如说新加一个字段,冗余保存完整消息的序列化,接着只GET这个字段就OK了。

本文来自 http://blog.sina.com.cn/s/blog_160965eee0102wflj.html

<think>我们被要求使用Redis实现消息队列。根据引用[1]和引用[2],Redis提供了三种方式:基于List结构、PubSub(发布订阅)和Stream结构。其中,引用[1]和引用[2]都提到,基于Stream的方式是比较完善的消息队列模型,推荐使用。 下面分别介绍这三种方式: ### 1. 基于List结构 Redis的List是一个双向链表,我们可以利用其`LPUSH`(或`RPUSH`)和`BRPOP`(或`BLPOP`)命令实现消息队列。 - 生产者使用`LPUSH`将消息放入队列的左侧(头部)。 - 消费者使用`BRPOP`从队列的右侧(尾部)阻塞地取出消息。 **优点**: - 实现简单,利用Redis已有的数据结构。 - 支持阻塞等待(使用`BRPOP`),避免消费者轮询。 **缺点**: - 不支持多消费者(一个消息只能被一个消费者消费)。 - 没有消息确认机制,如果消费者处理失败,消息就丢失了。 - 没有消息回溯功能。 #### 示例代码(生产者): ```java Jedis jedis = new Jedis("localhost"); // 向队列list1中放入消息 jedis.lpush("list1", "message1"); ``` #### 示例代码(消费者): ```java Jedis jedis = new Jedis("localhost"); // 阻塞地从队列list1中取出消息,0表示无限等待 List<String> result = jedis.brpop(0, "list1"); String message = result.get(1); ``` ### 2. 基于PubSub(发布订阅) Redis的发布订阅模式允许消息被多个消费者接收(广播)。 - 生产者发布消息到指定频道(channel)。 - 消费者订阅频道,接收消息。 **优点**: - 支持一对多的消息发布。 - 实时性较好。 **缺点**: - 消息是即时的,没有持久化。如果消费者不在线,消息就丢失了。 - 不支持消息的存储,因此无法回溯历史消息。 - 没有消息确认机制。 #### 示例代码(生产者): ```java Jedis jedis = new Jedis("localhost"); // 向频道channel1发布消息 jedis.publish("channel1", "message1"); ``` #### 示例代码(消费者): ```java Jedis jedis = new Jedis("localhost"); // 创建订阅对象 jedis.subscribe(new JedisPubSub() { @Override public void onMessage(String channel, String message) { // 处理消息 System.out.println("Received: " + message); } }, "channel1"); ``` ### 3. 基于Stream(推荐) Redis 5.0引入了Stream数据结构,它提供了更完善的消息队列功能: - 支持消息持久化。 - 支持消费者组(Consumer Group):多个消费者可以组成一个组,共同消费一个队列,每条消息只会被组内的一个消费者消费。 - 提供消息确认机制(ACK),确保消息被成功处理。 - 支持消息回溯。 #### 基本命令: - `XADD`:添加消息到流。 - `XREAD`:读取消息(可以阻塞)。 - `XGROUP`:创建消费者组。 - `XREADGROUP`:通过消费者组读取消息。 - `XACK`:确认消息已被处理。 #### 示例代码(生产者): ```java Jedis jedis = new Jedis("localhost"); // 向名为stream1的流中添加消息,消息包含字段key1和value1 jedis.xadd("stream1", StreamEntryID.NEW_ENTRY, Collections.singletonMap("key1", "value1")); ``` #### 示例代码(消费者组): 首先,创建消费者组: ```bash # 在Redis命令行中创建消费者组 XGROUP CREATE stream1 mygroup 0 ``` 然后,在Java中使用消费者组消费: ```java Jedis jedis = new Jedis("localhost"); // 消费者组mygroup中的消费者consumer1读取消息 Map.Entry<String, StreamEntryID> streamQuery = new AbstractMap.SimpleEntry<>("stream1", StreamEntryID.UNRECEIVED_ENTRY); List<Map.Entry<String, List<StreamEntry>>> result = jedis.xreadGroup("mygroup", "consumer1", 1, 0, // 读取1条消息,0表示阻塞直到有消息 streamQuery); // 处理消息 for (StreamEntry entry : result.get(0).getValue()) { System.out.println("Received: " + entry.getFields()); // 确认消息 jedis.xack("stream1", "mygroup", entry.getID()); } ``` ### 总结 - **List结构**:适用于简单的消息队列,单消费者,无持久化要求。 - **PubSub**:适用于广播消息,实时性要求高,但消息可丢失的场景。 - **Stream**:适用于需要持久化、多消费者组、消息确认等高级功能的场景,是最接近专业消息队列实现[^1][^2]。 在实际应用中,根据需求选择合适的方式。对于需要可靠消息队列的场景,推荐使用Stream结构。 ### 相关问题 1. Redis Stream 如何实现消息的持久化和重放? 2. 在 Redis 中,如何监控消息队列的运行状态? 3. Redis 作为消息队列与专业的消息队列(如 RabbitMQ)相比有哪些优缺点?</think>### Redis 实现消息队列的三种方式 Redis 提供了三种主要方式实现消息队列,根据引用[1][2][3]的内容整理如下: #### 1. 基于 List 结构(双向链表) ```bash # 生产者 LPUSH myqueue "message1" # 左侧插入消息 LPUSH myqueue "message2" # 消费者 BRPOP myqueue 0 # 阻塞式右侧弹出(0=无限等待) ``` **特点**: - 支持阻塞式弹出(`BRPOP`) - 消息持久化存储 - 简单高效,适合基本队列需求 - **缺点**:不支持多消费者组[^1][^2] #### 2. 基于 PubSub(发布订阅) ```bash # 消费者订阅 SUBSCRIBE channel1 # 生产者发布 PUBLISH channel1 "message" ``` **特点**: - 实时广播消息 - 支持多消费者 - **缺点**:消息不持久化(离线消费者丢失消息)[^1][^2] #### 3. 基于 Stream(推荐方式) ```bash # 生产者添加消息 XADD mystream * key1 "value1" key2 "value2" # 消费者组消费 XGROUP CREATE mystream mygroup 0 XREADGROUP GROUP mygroup consumer1 COUNT 1 STREAMS mystream > ``` **特点**: - 消息持久化存储 - 支持消费者组(同组内竞争消费) - 消息确认机制(ACK) - 支持消息回溯 - 提供阻塞读取能力[^1][^2] ### 完整 Java 实现示例(使用 Jedis + Stream) ```java import redis.clients.jedis.Jedis; import redis.clients.jedis.StreamEntry; import redis.clients.jedis.StreamEntryID; public class RedisStreamQueue { private static final String STREAM_KEY = "orders"; private static final String GROUP_NAME = "order_processor"; public static void main(String[] args) { // 初始化消费者组 initConsumerGroup(); // 启动生产者线程 new Thread(RedisStreamQueue::producer).start(); // 启动消费者线程 new Thread(RedisStreamQueue::consumer).start(); } private static void initConsumerGroup() { try (Jedis jedis = new Jedis("localhost")) { jedis.xgroupCreate(STREAM_KEY, GROUP_NAME, new StreamEntryID(), true); } } private static void producer() { try (Jedis jedis = new Jedis("localhost")) { int count = 1; while (true) { String order = "订单-" + count++; // 添加消息到流 jedis.xadd(STREAM_KEY, null, Map.of("order_id", order, "amount", String.valueOf(count * 10))); System.out.println("生产: " + order); Thread.sleep(1500); } } catch (Exception e) { e.printStackTrace(); } } private static void consumer() { try (Jedis jedis = new Jedis("localhost")) { while (true) { // 从消费者组读取消息 var result = jedis.xreadGroup( GROUP_NAME, "consumer-1", 1, 0, true, Map.of(STREAM_KEY, ">") ); if (!result.isEmpty()) { for (var stream : result) { for (var entry : stream.getValue()) { System.out.println("消费: " + entry.getFields()); // 处理成功后确认消息 jedis.xack(STREAM_KEY, GROUP_NAME, entry.getID()); } } } Thread.sleep(1000); } } catch (Exception e) { e.printStackTrace(); } } } ``` ### 三种方式对比 | **特性** | **List** | **PubSub** | **Stream** | |------------------|----------|------------|------------| | 消息持久化 | ✓ | ✗ | ✓ | | 阻塞式消费 | ✓ | ✗ | ✓ | | 多消费者组 | ✗ | ✗ | ✓ | | 消息确认(ACK) | ✗ | ✗ | ✓ | | 消息回溯 | ✗ | ✗ | ✓ | | 广播消息 | ✗ | ✓ | ✗ | | 适用场景 | 简单队列 | 实时通知 | 企业级队列 | ### 使用建议 1. **基本需求**:使用 List 结构(`LPUSH`+`BRPOP`) 2. **实时通知**:使用 PubSub(广播场景) 3. **企业级应用**:推荐使用 Stream 结构(支持消费者组和ACK)[^1][^2] Redis Stream 是目前最接近专业消息队列(如 Kafka/RabbitMQ)的实现方式,适合需要消息持久化、消费确认和多消费者组的场景[^3]。 ### 相关问题 1. Redis Stream 如何实现消息的持久化和重放? 2. 在 Redis 中,如何监控消息队列的运行状态? 3. Redis 作为消息队列与专业的消息队列(如 RabbitMQ)相比有哪些优缺点?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值