Redis(110)Redis的发布订阅机制如何使用?

Redis 的发布/订阅(Pub/Sub)机制允许消息的发布者和订阅者之间进行松耦合的消息传递。发布者将消息发布到特定的频道,而订阅者可以订阅该频道以接收消息。这种机制非常适合用于实时通知、消息广播等场景。

Redis 发布/订阅机制的基本使用

以下是 Redis 发布/订阅机制的详细使用方法和示例代码。

1. 启动 Redis 服务

确保 Redis 服务器正在运行。你可以通过命令行启动 Redis:

bash 复制代码
redis-server
2. 使用 Redis 客户端发布消息

可以使用 Redis 提供的客户端(如 Jedis)来发布和订阅消息。

发布者代码示例

java 复制代码
import redis.clients.jedis.Jedis;

public class RedisPublisher {
    public static void main(String[] args) {
        try (Jedis jedis = new Jedis("localhost")) {
            // 发布消息到频道 "my_channel"
            jedis.publish("my_channel", "Hello, Redis!");
            jedis.publish("my_channel", "Another message.");
        }
    }
}

订阅者代码示例

java 复制代码
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;

public class RedisSubscriber {
    public static void main(String[] args) {
        try (Jedis jedis = new Jedis("localhost")) {
            // 创建订阅者
            JedisPubSub jedisPubSub = new JedisPubSub() {
                @Override
                public void onMessage(String channel, String message) {
                    System.out.println("Received message: " + message + " from channel: " + channel);
                }
            };

            // 订阅频道 "my_channel"
            jedis.subscribe(jedisPubSub, "my_channel");
        }
    }
}

3. 运行发布者和订阅者

  • 首先运行订阅者代码,订阅者将会等待接收来自 my_channel 频道的消息。
  • 然后运行发布者代码,发布者将消息发布到 my_channel 频道。订阅者将接收到并打印这些消息。

4. 详细示例:多个发布者和多个订阅者

发布者示例
java 复制代码
import redis.clients.jedis.Jedis;

public class RedisPublisher {
    public static void main(String[] args) {
        try (Jedis jedis = new Jedis("localhost")) {
            for (int i = 0; i < 5; i++) {
                String message = "Message " + i;
                System.out.println("Publishing message: " + message);
                jedis.publish("news_channel", message);
                try {
                    Thread.sleep(1000); // 等待1秒
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
订阅者示例
java 复制代码
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;

public class RedisSubscriber {
    public static void main(String[] args) {
        try (Jedis jedis = new Jedis("localhost")) {
            // 创建订阅者
            JedisPubSub jedisPubSub = new JedisPubSub() {
                @Override
                public void onMessage(String channel, String message) {
                    System.out.println("Received message: " + message + " from channel: " + channel);
                }

                @Override
                public void onSubscribe(String channel, int subscribedChannels) {
                    System.out.println("Subscribed to channel: " + channel);
                }

                @Override
                public void onUnsubscribe(String channel, int subscribedChannels) {
                    System.out.println("Unsubscribed from channel: " + channel);
                }
            };

            // 订阅频道 "news_channel"
            jedis.subscribe(jedisPubSub, "news_channel");
        }
    }
}

5. 使用模式匹配订阅

除了订阅特定的频道外,Redis 还支持模式匹配订阅。模式订阅者可以订阅符合特定模式的所有频道。

模式订阅示例
java 复制代码
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;

public class RedisPatternSubscriber {
    public static void main(String[] args) {
        try (Jedis jedis = new Jedis("localhost")) {
            // 创建模式订阅者
            JedisPubSub jedisPubSub = new JedisPubSub() {
                @Override
                public void onPMessage(String pattern, String channel, String message) {
                    System.out.println("Received message: " + message + " from channel: " + channel + " matching pattern: " + pattern);
                }

                @Override
                public void onPSubscribe(String pattern, int subscribedChannels) {
                    System.out.println("Subscribed to pattern: " + pattern);
                }

                @Override
                public void onPUnsubscribe(String pattern, int subscribedChannels) {
                    System.out.println("Unsubscribed from pattern: " + pattern);
                }
            };

            // 订阅匹配 "news.*" 模式的所有频道
            jedis.psubscribe(jedisPubSub, "news.*");
        }
    }
}

6. 发布订阅注意事项

  1. 消息丢失:Redis Pub/Sub 机制不保证消息的持久化。如果没有订阅者在线,消息将被丢弃。
  2. 消息顺序:消息按照发布的顺序被订阅者接收,但如果有多个订阅者,顺序不保证在所有订阅者之间一致。
  3. 性能:对于高流量或需要消息持久化的场景,可以考虑使用 Redis Streams 或其他消息队列系统(如 Kafka、RabbitMQ)。

7. 高级示例:多线程环境下的发布订阅

多线程发布者
java 复制代码
import redis.clients.jedis.Jedis;

public class MultiThreadRedisPublisher implements Runnable {
    private String channel;
    private String message;

    public MultiThreadRedisPublisher(String channel, String message) {
        this.channel = channel;
        this.message = message;
    }

    @Override
    public void run() {
        try (Jedis jedis = new Jedis("localhost")) {
            jedis.publish(channel, message);
            System.out.println("Published message: " + message + " to channel: " + channel);
        }
    }

    public static void main(String[] args) {
        Thread publisher1 = new Thread(new MultiThreadRedisPublisher("channel1", "Hello from publisher 1"));
        Thread publisher2 = new Thread(new MultiThreadRedisPublisher("channel2", "Hello from publisher 2"));

        publisher1.start();
        publisher2.start();
    }
}
多线程订阅者
java 复制代码
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;

public class MultiThreadRedisSubscriber implements Runnable {
    private String channel;

    public MultiThreadRedisSubscriber(String channel) {
        this.channel = channel;
    }

    @Override
    public void run() {
        try (Jedis jedis = new Jedis("localhost")) {
            JedisPubSub jedisPubSub = new JedisPubSub() {
                @Override
                public void onMessage(String channel, String message) {
                    System.out.println("Received message: " + message + " from channel: " + channel);
                }
            };

            jedis.subscribe(jedisPubSub, channel);
        }
    }

    public static void main(String[] args) {
        Thread subscriber1 = new Thread(new MultiThreadRedisSubscriber("channel1"));
        Thread subscriber2 = new Thread(new MultiThreadRedisSubscriber("channel2"));

        subscriber1.start();
        subscriber2.start();
    }
}

通过这些示例代码,展示了如何使用 Redis 的发布/订阅机制,包括基本的发布和订阅、模式匹配订阅以及多线程环境下的发布和订阅。Redis 的发布/订阅机制非常强大,可以用于各种实时消息传递的场景。

相关推荐
m0_4711996312 小时前
【小程序】订单数据缓存 以及针对海量库存数据的 懒加载+数据分片 的具体实现方式
前端·vue.js·小程序
编程大师哥12 小时前
Java web
java·开发语言·前端
Murrays12 小时前
【React】01 初识 React
前端·javascript·react.js
oMcLin12 小时前
如何在 Debian 10 上通过配置 Redis 集群的持久化选项,提升高可用性缓存系统的容错性与性能?
redis·缓存·debian
大喜xi12 小时前
ReactNative 使用百分比宽度时,aspectRatio 在某些情况下无法正确推断出高度,导致图片高度为 0,从而无法显示
前端
helloCat12 小时前
你的前端代码应该怎么写
前端·javascript·架构
电商API_1800790524712 小时前
大麦网API实战指南:关键字搜索与详情数据获取全解析
java·大数据·前端·人工智能·spring·网络爬虫
康一夏12 小时前
CSS盒模型(Box Model) 原理
前端·css
web前端12312 小时前
React Hooks 介绍与实践要点
前端·react.js
我是小疯子6612 小时前
JavaScriptWebAPI核心操作全解析
前端