redis中的发布订阅(Pub/Sub)

这里使用nodejs的redis模块说明,具体可见https://www.npmjs.com/package/redis,先来通过一个简单的例子了解下redis中的Pub/Sub具体怎么实现吧。。

var express = require(‘express‘);
var router = express.Router();
var redis = require("redis");
 /* GET home page. */
router.get(‘/‘, function(req, res, next) {

  var client1 = redis.createClient();
  var client2  = redis.createClient(6379,"192.168.20.132");
  var msg_count = 0;

  client1.on("error", function (err) {
    console.log("Error " + err);
  });

  client2.on("error",function(err){
    console.log(err);
  });

  client1.on("subscribe", function (channel, count) {     //channel为订阅的事件类型(通道),count为当前客户端订阅的channel总数
    client2.publish("a nice channel", "I am sending a message.");
    client2.publish("a nice channel", "I am sending a second message.");
    client2.publish("a nice channel", "I am sending my last message.");
    console.log("client1 sub count:" + count);
    console.log("client1 sub channel:" + channel);
  });

  client1.on("message", function (channel, message) {     //channel为send端所发送的订阅类型,message为该channel中所收到的信息
    console.log("client1 channel name: " + channel + "->message: " + message);
    msg_count += 1;
    if (msg_count === 3) {
      client1.unsubscribe();    //退订所有channel
      client1.end();    //client1退出Pub/Sub模式,可继续执行其它redis命令
      client2.end();
    }
  });
  client1.subscribe("a nice channel");  //自动触发"sunscribe"事件
  res.render(‘index‘, {title: ‘Express‘});
});
 module.exports = router;

将上述代码保存为index.js文件,替换Express项目中的index.js,然后启动后浏览器中输入:localhost:3000,观察Webstorm中打印信息如下:

client1 sub count:1
client1 sub channel:a nice channel
client1 channel name: a nice channel->message: I am sending a message.
client1 channel name: a nice channel->message: I am sending a second message.
client1 channel name: a nice channel->message: I am sending my last message.

这里来说明下:

  • require来redis模块后,通过redis的createClient方法创建 redis客户端连接,该方法可以指定三个参数,分别为:连接的redis server端口、server ip及可配置的options。不带任何参数时默认连接本机redis server的6379端口,编码时也可使用createClient(6379,"ip address",{})的options为空的形式使用默认配置,当然也可直接省略options对象
  • error事件为client端操作报错时自动触发的事件
  • subscribe事件和message事件稍后说明

发布订阅?

redis中的发布订阅,自我的理解是:发布订阅就是有一端发布消息,一端订阅消息即接收消息,这里的发布订阅端都可以称为client端,也就是说一个client既可以发布多个消息,亦可以订阅多个消息。

说 到消息,到底什么是消息?redis中的每条消息是一条带有三个元素的多条批量回复(multi-bulk-reply)。这货刚听时候着实难以理解,下 面继续。这里的第一个元素是消息类型,redis中消息类型并非我们理解的String、Object等,而是subscribe、 unsubscribe、message等类型。

  • subscribe:如果类型为subscribe,则表示当前客户端成功订阅 了第二个元素所示频道(频道可以理解为消息的名称或channel,因为redis中client端发布消息后,redis server端再根据其它客户端是否订阅该名称或channel而转发该消息至订阅端),这时信息的第三个元素则记录了目前客户端已订阅频道的总数
  • unsubscribe:表示当前客户端成功地退订了信息第二个元素所指示的频道。 信息的第三个元素记录了客户端目前仍在订阅的频道数量。 当客户端订阅的频道数量降为 0 时, 客户端不再订阅任何频道, 它可以像往常一样, 执行任何 Redis 命令
  • message:表示这条信息是由某个客户端执行 PUBLISH命令所发送的, 真正的信息。 信息的第二个元素是信息来源的频道, 而第三个元素则是信息的内容

如果客户端执行以下命令:

redis> SUBSCRIBE first second

表明该客户端订阅了名称为first和second的两个channel,那么它将收到以下回复:

1) "subscribe"
2) "first"
3) (integer) 1

1) "subscribe"
2) "second"
3) (integer) 2

其中,subscribe表明此消息类型,first为channel名称,(integer) 1表示当前客户端订阅的channel总数为1,如果此时另一个客户端执行了如下命令:

redis> PUBLISH second Hello

表示该客户端发布了一个channel为second,内容为hello的一条信息,那么之前订阅了channel为second的客户端将收到以下信息:

1) "message"
2) "second"
3) "hello"

其中,message表明该消息为另一客户端发送而来,second为channel名,此时hello为信息内容。如果这时订阅的客户端再执行以下命令:

redis> UNSUBSCRIBE

表明订阅者退订所有之前订阅的channel,这时命令执行后收到回复如下:
1) "unsubscribe"
2) "second"
3) (integer) 1

1) "unsubscribe"
2) "first"
3) (integer) 0

可以看到消息退订时是一条条挨个退订,而且是“后订先退”顺序,当然也可以直接指定退订的channel。

有一点觉得比较重要:

通过PUBLISH发出去的信息,是不会保存在服务端的,服务端只是做中转处理。也就是说,如果客户端PUBLISH了信息至channel,而没有其它客户端订阅该channel,那么该消息也是无效的,消息的传递是以channel为载体的。

时间: 2024-11-05 19:05:51

redis中的发布订阅(Pub/Sub)的相关文章

Redis中的发布与订阅

redis中实现发布与订阅相对于zookeeper非常简单.直接使用publish和subscribe就行. subscrible news; 订阅news这个channel publish news  a-dog-eat-a-pig 在news这个channel就接受到信息

JavaScript实现的发布/订阅(Pub/Sub)模式

JavaScript实现的发布/订阅(Pub/Sub)模式时间 2016-05-02 18:47:58  GiantMing's blog原文  http://giantming.net/javascriptshi-xian-de-fa-bu-ding-yue-pub-sub-mo-shi/主题 JavaScript 观察者模式前段时间看了一下发布订阅者模式(也叫观察者模式),今天看<基于mvc的JavaScript的富应用开发>又看到了它,这个设计模式是非常有用的,正好写篇博客来分享一下.(

Javascript中理解发布--订阅模式

阅读目录 发布订阅模式介绍 如何实现发布--订阅模式? 发布---订阅模式的代码封装 如何取消订阅事件? 全局--发布订阅对象代码封装 理解模块间通信 回到顶部 发布订阅模式介绍 发布---订阅模式又叫观察者模式,它定义了对象间的一种一对多的关系,让多个观察者对象同时监听某一个主题对象,当一个对象发生改变时,所有依赖于它的对象都将得到通知. 现实生活中的发布-订阅模式: 比如小红最近在淘宝网上看上一双鞋子,但是呢 联系到卖家后,才发现这双鞋卖光了,但是小红对这双鞋又非常喜欢,所以呢联系卖家,问卖

Redis七(发布订阅)

发布与订阅(pub/sub) 介绍 Redis 通过 PUBLISH . SUBSCRIBE 等命令实现了订阅与发布模式, 这个功能提供两种信息机制, 分别是订阅/发布到频道和订阅/发布到模式 订阅者可以订阅一个或多个频道,发布者向一个频道发送消息后,所有订阅这个频道的订阅者都将收到消息,而发布者也将收到一个数值,这个数值是收到消息的订阅者的数量.订阅者只能收到自它开始订阅后发布者所发布的消息,而之前发布的消息是收不到的. 运行原理: Redis 的 SUBSCRIBE 命令可以让客户端订阅任意

redis 的消息发布订阅

redis支持pub/sub功能(可以用于消息服务器),这个功能类似mq,这里做一个简单的介绍 Pub/Sub Pub/Sub 从字面上理解就是发布(Publish)与订阅(Subscribe),在Redis中,你可以设定对某一个key值进行消息发布及消息订阅, 当一个key值上进行了消息发布后,所有订阅它的客户端都会收到相应的消息.这一功能最明显的用法就是用作实时消息系统,比如普通的即时聊天,群聊等功能. 客户端1:subscribe rain 客户端2:PUBLISH rain "hello

redis事务以及发布订阅

1. 什么是redis的事务? ???????? redis事务就是一个命令执行的队列,将一系列预定义命令包装成一个整体,就是一个队列.当执行的时候,一次性按照添加顺序依次执行,中间不会被打断或者干扰. 2. 能干嘛? ???????? 一个队列中,一次性,顺序性,排他性的执行一系列命令 3. redis事务基本操作 开启事务:multi 设置事务的开始位置,这个指令开启后,后面所有的指令都会加入事务中 执行事务: exec 设置事务的结束位置,同时执行事务,与multi成对出现,成对使用 取消

OrcharNoCMS中的发布订阅使用

对于Orchard里面的EventBus,没有太多的文章去介绍说明.它最好的应用是发布订阅的应用. 使用介绍: 在Car模块中,我们定义一个接口,继承IEventHandler接口. 当我们在创建一条记录时,去执行所有该接口的实现. 接下来是实现该接口: 这里是一个验证的方法.当前台post过来的model里面Price<0,我们就抛出异常,ValidationException. 在Service中如何使用Handler呢? 执行Create方法时,从容器中找到所有的ICarEventHand

【Redis】jedis客户端实现redis消息的发布订阅(实时消息中间件)

发布 package com.chiwei.redis; import java.io.BufferedReader; import java.io.InputStreamReader; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import redis.clients.jedis.Jedis; public class RedisPublisher { private static final Logger log = L

C# Redis系列(三)-Redis发布订阅及客户端编程

发布订阅模型 Redis中的发布订阅 客户端编程示例 0.3版本Hredis 发布订阅模型 在应用级其作用是为了减少依赖关系,通常也叫观察者模式.主要是把耦合点单独抽离出来作为第三方,隔离易变化的发送方和接收方. 发送方:只负责向第三方发送消息.(杂志社把读者杂志交给邮局) 接收方:被动接收消息.(1:向邮局订阅读者杂志,2:门口去接邮过来的杂志) 第三方作用是:存储订阅杂志的接收方,并在杂志过来时送给接收方. (邮局) C#示例,发送方把杂志放到邮局里面: if (QA.AddBug()) E