前言

使用过简单的python的ZMQ:server开启3个线程,client发送心跳包,如果服务端超过n秒没应答,则重新连接。

网上找的案例,server使用的zmq.device,但是一直不明白什么含义。

案例链接:http://nphard.me/2016/03/05/pyzmq-demo/

client使用了超时重连,传输数据时,如果server没有响应则重新连接并重新发送数据,这样会导致,客户端重复发送多条数据,被服务端接收处理,但是服务端并没有回复。(以下准备解释,但是没成功)

以下为参考网站,很多英文的看不懂。

http://api.zeromq.org/

这里是翻译的目录

http://pyzmq.readthedocs.io/en/latest/

http://zguide.zeromq.org/py:all

这里搜到的中文教材

https://learning-0mq-with-pyzmq.readthedocs.io/en/latest/index.html

https://wizardforcel.gitbooks.io/zmq-guide/content/chapter1.html

具体细节参考的中文资料在对应目录内给出。

zmq的router和dealer是什么

网上有很多资料介绍zmq的几种模式,以下介绍req/rep应答模式。

首先确认一个概念:XREQ/XREP are aliases for ROUTER/DEALER. XREQ/XREP were used in ZeroMQ 2.x.

那么router和dealer是什么?直接上链接:

http://www.cnblogs.com/fengbohello/p/4743868.html

官网介绍了应答模式的几种区别:http://zguide.zeromq.org/py:chapter3

以下是自己的理解,如果有问题请留言,谢谢。

参考画图:https://stackoverflow.com/questions/23581172/what-is-zmq-router-and-zmq-dealer-in-python-zeromq

zmq的device

参考官方api:http://pyzmq.readthedocs.io/en/latest/api/zmq.devices.html

参考device详细介绍:http://pjwqq.iteye.com/blog/2260254

根据上图给出代码,运行device_func,frontend面向客户端(发起请求方),绑定router接收请求,device内部通过backend与服务端连接,dealer向服务端发起请求,并将数据传达给server。

所以流程就是这样:client (req)  ----->  router/dealer (queue)  ----->  server (rep)

device 和 server 端:通过多线程写在一起的。运行先后顺序是  1、device  ---  2、server / client (随便哪一个先启动都会等待后启动连接,如果中途server异常,client需要超时重试, !!!!!^~^)

def device_func():
thread_num = 3
context = zmq.Context()
url_router = 'inproc://ping-server'
url_dealer = 'tcp://*:5559' # Socket do cliente
frontend = context.socket(zmq.ROUTER) # 或者是 zmq.ROUTER zmq.XREP
frontend.bind(url_dealer) # Socket do servidor
backend = context.socket(zmq.DEALER) # 或者是 zmq.DEALER zmq.XREQ
backend.bind(url_router) for i in range(1, thread_num + 1):
thread = threading.Thread(target=server_func, args=(i, url_router, context))
thread.start() # property找不到 Device()报错
zmq.device(zmq.QUEUE, frontend, backend) frontend.close()
backend.close()
context.term() def server_func(name, url_router, context):
print(">>> start %d %s" % (name, '.' * 50))
socket = context.socket(zmq.REP)
socket.connect(url_router)
while True:
try:
message = socket.recv()
r_data = msgpack.unpackb(message, encoding='utf-8')
print('server %d received:' % name, r_data) data = 'server %d send: %s' % (name, utils.time_now())
print(data)
s_data = msgpack.packb(data)
socket.send(s_data) except:
traceback.print_exc()
socket.close()
break

client端,连接device的frontend端(dealer):

def client2():
url_dealer = 'tcp://localhost:5559'
context = zmq.Context()
socket = context.socket(zmq.REQ) print("Collecting data from server…")
socket.connect(url_dealer)
pid = os.getpid()
i = 0
while 1:
i += 1
# data = input('\n>>:').strip()
data = ('send - msg%d' % i)
# print(data)
s_data = msgpack.packb(data.encode())
socket.send(s_data) ret = socket.recv()
r_data = msgpack.unpackb(ret, encoding='utf-8')
print('>>>received msg%d server info: %s \n' % (i, r_data))

以上一起理解了zmq的device、router、dealer。

官网指出:function  zmq.device(device_typefrontendbackend) Deprecated since version libzmq-3.2: Use zmq.proxy 取而代之的 zmq.proxy(frontendbackendcapture)

同时又有 class zmq.devices.Device(device_type=3in_type=Noneout_type=None) 测试了跑不起来

备注:

inproc 是zmq本地 进程/线程 的传输方式。可以自定义。

 zmq的Polling

主要功能应该是 一个程序种需要建立多个socket连接时,需要接收数据,又要转发数据等。参考以下链接:

官方文档

http://pjwqq.iteye.com/blog/2260791

官方API翻译的

我这里主要是使用到client的超时重试功能。重现客户端不断超时重试发送数据,服务端不断接收回复。数据大量冗余重复。

server:同上,不过加了一句 sleep:

time.sleep(6)
message = socket.recv()

client:

def conn_socket(url_dealer):
""" 建立socket连接"""
try: context = zmq.Context()
socket = context.socket(zmq.REQ)
socket.connect(url_dealer) poller = zmq.Poller()
poller.register(socket, zmq.POLLIN) print('[%d] socket conn success' % os.getpid())
return socket, poller except:
traceback.print_exc() def close_socket(socket, poller):
""" 关闭socket连接"""
try:
socket.close()
poller.unregister(socket)
except:
traceback.print_exc() def client1():
""" """
url_dealer = 'tcp://localhost:5559'
socket, poller = conn_socket(url_dealer)
try: i = 0
while 1:
i += 1
# data = input('\n>>:').strip()
data = 'msg%d' % i
print('send:%s' % data)
# print(data)
s_data = msgpack.packb(data.encode())
socket.send(s_data) while True:
# 超时后重新连接,参数是毫秒
if poller.poll(5 * 1000):
ret = socket.recv()
r_data = msgpack.unpackb(ret, encoding='utf-8')
print('>>>received:%s' % r_data)
break
else:
close_socket(socket, poller)
socket, poller = conn_socket(url_dealer) print('*resend:%s' % data)
socket.send(s_data) except Exception as e:
traceback.print_exc()
finally:
close_socket(socket, poller)
pass

注意看:我故意将server的sleep值 > client 的超时重试的值,那么问题就重现了:

客户端不停的连接重试重发数据:

服务端不停的接收回复:

看了官方文档的后续的更多的高级功能介绍,发现并没有找到解决方案,暂时智能把超时时间设置的足够大一些,并在返回数据中添加一个状态,重试1次加一个1,方便反查和监控报警。

pyzmq学习笔记的更多相关文章

  1. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  2. PHP-自定义模板-学习笔记

    1.  开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2.  整体架构图 ...

  3. PHP-会员登录与注册例子解析-学习笔记

    1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...

  4. 2014年暑假c#学习笔记目录

    2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...

  5. JAVA GUI编程学习笔记目录

    2014年暑假JAVA GUI编程学习笔记目录 1.JAVA之GUI编程概述 2.JAVA之GUI编程布局 3.JAVA之GUI编程Frame窗口 4.JAVA之GUI编程事件监听机制 5.JAVA之 ...

  6. seaJs学习笔记2 – seaJs组建库的使用

    原文地址:seaJs学习笔记2 – seaJs组建库的使用 我觉得学习新东西并不是会使用它就够了的,会使用仅仅代表你看懂了,理解了,二不代表你深入了,彻悟了它的精髓. 所以不断的学习将是源源不断. 最 ...

  7. CSS学习笔记

    CSS学习笔记 2016年12月15日整理 CSS基础 Chapter1 在console输入escape("宋体") ENTER 就会出现unicode编码 显示"%u ...

  8. HTML学习笔记

    HTML学习笔记 2016年12月15日整理 Chapter1 URL(scheme://host.domain:port/path/filename) scheme: 定义因特网服务的类型,常见的为 ...

  9. DirectX Graphics Infrastructure(DXGI):最佳范例 学习笔记

    今天要学习的这篇文章写的算是比较早的了,大概在DX11时代就写好了,当时龙书11版看得很潦草,并没有注意这篇文章,现在看12,觉得是跳不过去的一篇文章,地址如下: https://msdn.micro ...

随机推荐

  1. HAOI2019+十二省联考 游记

    Day1 T1 考前还奶了一口不会考01Trie的,也就没有学,然后60分BOOM T2 不会SAM,告辞,30分滚粗 T3 传统实现题答?2p,2u,2g分别对应素数,莫比乌斯函数,原根?没看出来, ...

  2. Glog使用记录

    1.Flag_xxx FLAGS_logtostderr = false; //是否将所有日志输出到stderr,而非文件 FLAGS_alsologtostderr = false; //日志记录到 ...

  3. Linux下nc命令的使用

    nc命令的作用 实现任意TCP/UDP端口的侦听,nc可以作为server以TCP或UDP方式侦听指定端口 端口的扫描,nc可以作为client发起TCP或UDP连接 机器之间传输文件 机器之间网络测 ...

  4. ZooKeeper-API 监听

    以服务动态上下线通知为例 Client 监听服务器状态 public class DistributeClient { private String connectString = "127 ...

  5. 使用Docker安装ELK系列(超简单)

    root权限 docker版本:1.13.1 ELK版本:6.4.3 项目中均关闭X-Pack 一.安装Elasticsearch 新建elasticsearch目录,并再其下新建文件config/e ...

  6. 实验一 Java环境的搭建&Eclipse的安装

    本次实验为在自己电脑上搭建Java环境,熟悉Java的编译和运行环境并安装Eclipse 一.JAVA环境的搭建 1.从Oracle网站上下载Java的jdk [https://www.oracle. ...

  7. PMP知识点(六)——项目经理权利类型

    社会心理学家约翰.弗伦奇(John.French)和伯特伦.雷文(Bertram.Raven)在他们1959年发表的一篇迄今不朽的论文中,提出了五种权力类别的模型,这个模型折射出权力拥有者依赖的不同基 ...

  8. js循环内0.5s停止

    var time_begin = new Date().getTime(); while(true){ if(new Date().getTime()-time_begin > 500) bre ...

  9. input file 重复上传同一张图片失效的解决办法

    解决方法: 每次取消图片预览后,重置input[type=’file’]的value的值 链接:https://blog.csdn.net/zd717822023/article/details/78 ...

  10. Contest2162 - 2019-3-28 高一noip基础知识点 测试5 题解版

    传送门 T1 单调栈 按照b排序 在家每一个物品时,判断一下a和b的关系 如果s[sta[top]].a>=s[i].b,就弹栈 记录所有时候的height,并取最大值 T2 单调栈裸题 单调栈 ...