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 的发布/订阅机制非常强大,可以用于各种实时消息传递的场景。

相关推荐
bearpping3 小时前
Nginx 配置:alias 和 root 的区别
前端·javascript·nginx
@大迁世界4 小时前
07.React 中的 createRoot 方法是什么?它具体如何运作?
前端·javascript·react.js·前端框架·ecmascript
January12074 小时前
VBen Admin Select 选择框选中后仍然显示校验错误提示的解决方案
前端·vben
. . . . .4 小时前
前端测试框架:Vitest
前端
xiaotao1314 小时前
什么是 Tailwind CSS
前端·css·css3
战南诚5 小时前
VUE中,keep-alive组件与钩子函数的生命周期
前端·vue.js
发现一只大呆瓜5 小时前
React-彻底搞懂 Redux:从单向数据流到 useReducer 的终极抉择
前端·react.js·面试
霍理迪5 小时前
Vue的响应式和生命周期
前端·javascript·vue.js
李剑一6 小时前
别再瞎写了!Cesium 模型 360° 环绕,4 套源码全公开,项目直接用
前端