NetMQ发布订阅C#示例
NetMQ (ZeroMQ to .Net),ØMQ号称史上最快中间件。它对socket通信进行了封装,使得我们不需要写socket函数调用就能完成复杂的网络通信。和一般意义上的消息队列产品不同的是,它没有消息队列服务器,而更像是一个网络通信库。从网络通信的角度看,它处于会话层之上,应用层之下。【ZeroMQ 官网】:http://zeromq.org
ØMQ有4个基本通信模型:分别是一对一结对模型(Exclusive-Pair)、请求回应模型(Request-Reply)、发布订阅模型(Publish-Subscribe)、推拉模型(Push-Pull)。
Request-reply pattern 请求-回复模型
- 这种模型主要用于从客户端向一个或多个服务实例发送请求,然后等待紧接着对于每个请求的回复
- 里面又具体分了ZMQ_REQ ZMQ_REP ZMQ_DEALER ZMQ_ROUTER
- REQ 发送完消息后,必须接收一个回应消息后,才能发送新的消息
- REP当接收消息时,都会返回一个消息
Publish-subscribe pattern 发布-订阅模式
- 这种模式主要用于1对多的数据发布(一个发布者,多个订阅者)
- 里面又具体分了ZMQ_PUB ZMQ_SUB
- PUB发送消息给所有的SUB。如果此时SUB没有启动,下次启动时会漏掉该消息
Pipeline pattern 管道模式
- 这种模式主要用于发布数据到由管道排列的节点上面,数据总是沿着管道流动。每个管道阶段连接了至少一个节点
- 里面又具体分了ZMQ_PUSH ZMQ_PULL
- 一个1对N队列的实现,PUSH将数据放入队列,PULL从队列中不取出数据。数据会负载均衡的发送给每一个PULL
Exclusive pair pattern 独立对模式
- peer to peer 模式。主要用于进程内部线程间通信
- 里面又具体分了ZMQ_PAIR
- 线程间1-to-1队列的实现,采用了lock free实现,所以速度很快
下面是订阅发布的示例代码:
发布服务端:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 | publicstaticclassNetMQPub  {      readonlystaticManualResetEvent _terminateEvent = newManualResetEvent(false);      /// <summary>      /// NetMQ 发布模式      /// </summary>      publicstaticvoidStart()      {          string[] wethers = newstring[5] {"晴朗","多云","阴天","小雨","暴雪"};          //CTRL+C 退出程序          Console.CancelKeyPress += Console_CancelKeyPress;          Console.WriteLine("发布多个地区天气预报:");          using(varcontext = NetMQContext.Create())          {              using(varpublisher = context.CreatePublisherSocket())              {                  varrng = newRandom();                  stringmsg;                  intsleeptime = 10;                  while(_terminateEvent.WaitOne(0) == false)                  {                      //随机生成天气数据                      intzipcode = rng.Next(0, 99);                      inttemperature = rng.Next(-50, 50);                      intwetherId = rng.Next(0, 4);                      msg = string.Format("{0} {1} {2}", zipcode, temperature, wethers[wetherId]);                      publisher.Send(msg,Encoding.UTF8, zmq.SendReceiveOptions.DontWait);                      Console.WriteLine(msg);                      Thread.Sleep(sleeptime);                  }              }          }      }      staticvoidConsole_CancelKeyPress(objectsender, ConsoleCancelEventArgs e)      {          Console.WriteLine("exit...");          _terminateEvent.Set();      }  } | 
订阅客户端,可启动多个实例来模拟接收天气信息:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 | publicstaticclassNetMQSub  {      publicdelegatevoidGetDataHandler(stringmessage);      publicstaticeventGetDataHandler OnGetData;      /// <summary>      /// NetMQ 订阅模式      /// </summary>      publicstaticvoidStart()      {                     varrng = newRandom();          intzipcode = rng.Next(0, 99);          Console.WriteLine("接收本地天气预报 {0}...", zipcode);          OnGetData += newGetDataHandler(ProcessData);            using(varcontext = NetMQContext.Create())          using(varsubscriber = context.CreateSubscriberSocket())          {              subscriber.Subscribe(zipcode.ToString(CultureInfo.InvariantCulture));              while(true)              {                  stringresults = subscriber.ReceiveString(Encoding.UTF8);                  Console.Write(".");                  string[] split = results.Split(new[] { ' '}, StringSplitOptions.RemoveEmptyEntries);                  intzip = int.Parse(split[0]);                  if(zip == zipcode)                  {                      OnGetData(results);                  }                               }          }      }      publicstaticvoidProcessData(stringmsg)      {          Console.WriteLine("天气情况:"+ msg);      }  } | 
 
 
 
                
            
         浙公网安备 33010602011771号
浙公网安备 33010602011771号