- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有以下一段 C++ 代码,它打开 ZMQ 订阅者套接字并在无限循环中接收消息。
listener.cc
:(代码应该可以工作,编译:g++ -lzmq listener.cc
)
#include <iostream>
#include <zmq.hpp>
class Listener {
public:
Listener() {
std::cout << "constructor call" << std::endl;
// Not working:
// zmq::context_t context(1);
// sck = new zmq::socket_t(context, ZMQ_SUB);
// sck->connect("tcp://127.0.0.1:9999");
// sck->setsockopt( ZMQ_SUBSCRIBE, "", 0);
std::cout << "constructor end" << std::endl;
}
void run() {
// Ok:
zmq::context_t context(1);
sck = new zmq::socket_t(context, ZMQ_SUB);
sck->connect("tcp://127.0.0.1:9999");
sck->setsockopt(ZMQ_SUBSCRIBE, "", 0);
while (1) { // Receive messages, not interesting:
std::cout << "listening..." << std::endl;
zmq::message_t message;
sck->recv(&message);
std::cout << "received something" << std::endl;
}
}
zmq::socket_t *sck;
};
int main(int argc, char *argv[]) {
Listener listener;
std::cout << "foo" << std::endl;
listener.run();
return 0;
}
到目前为止,代码按预期工作:
$ g++ -lzmq listener.cc
$ ./a.out
constructor call
constructor end
foo
listening...
不过,我想把zmq-context/socket的初始化移到类的构造函数中(被注释掉的部分)。但是随后代码根本不会从构造函数调用中返回,构造函数中的所有语句都被执行但是 main
的第二行没有被执行并且程序卡住了。输出是:
$ g++ -lzmq listener.cc
$ ./a.out
constructor call
constructor end
我唯一想到的就是主线程因为某种原因停止执行。谁能解释一下并提供解决方案?
干杯
最佳答案
ZeroMQ 使用每个 Context( nIOthreads = 1 )
实例作为引擎盖下的一个非常强大的引擎,必须小心谨慎,这样资源管理过程就不会让人感到意外(因为阻塞/卡住就是这样一种情况)。
如果有一些活跃使用的套接字实例(在 Context() 实例的幕后管理),可能会出现这样一种情况,即在进入析构函数处理阶段之前并非所有传输都已完成,或者如果有人手动执行类似的步骤,试图尝试 .close()
这样的套接字实例和/或 .term()
上下文实例。
无论有意或无意,人们都会多次陷入这种困境。
ZeroMQ native API 文档在这个主题上非常清楚,并警告了一个风险,即尚未完成的低级事务可能会让代码无限等待永远不会出现的外部(远程代理操作)事件。这种无意识的代码看起来像是卡住/挂起的故障,但只是由于没有意识到这种风险并且没有采取适当的预防措施而导致这种确定性的情况。
虽然较新的 API 版本更改了一些默认设置,但我建议所有用户明确设置安全配置,即使较新的默认设置可能避免手动执行此操作的需要。然而,这种做法有助于提高人们对应该及时考虑哪些类型的碰撞的认识 distributed-system设计实践。
.setsockopt( ZMQ_LINGER, 0 );
zmq_term()
shall block until the following conditions are satisfied:All sockets open within context have been closed with
zmq_close()
.
For each socket within context, all messages sent by the application withzmq_send()
have either been physically transferred to a network peer, or the socket's linger period set with theZMQ_LINGER
socket option has expired.
如上所述,这是每个套接字实例化的经验法则。
class Listener {
// zmq::context_t aClassLocalCONTEXT; // MAY GET SET LOCAL CTX BY VALUE
// zmq::socket_t aClassLocalSOCKET; // MAY GET SET LOCAL SCK BY VALUE EITHER
zmq::socket_t *sck;
public:
Listener() {
std::cout << "constructor call" << std::endl;
// zmq::context_t context(1); // not a best practice here
// ---------------------------------------------------
// sck = new zmq::socket_t( aClassLocalCONTEXT, ZMQ_SUB );
sck = new zmq::socket_t( context, ZMQ_SUB );
sck->setsockopt( ZMQ_LINGER, 0 ); // ALWAYS, best before .bind()/.connect()
sck->connect( "tcp://127.0.0.1:9999" );
sck->setsockopt( ZMQ_SUBSCRIBE, "", 0 );
// ----------------------------------------------// IF SETUP BY AN INSTANTIATION CALL INTERFACE
// aClassLocalSOCKET->setsockopt( ZMQ_LINGER, 0 );
// aClassLocalSOCKET->connect( ... );
// aClassLocalSOCKET->setsockopt( ZMQ_SUBSCRIBE, "", 0 );
// ---------------------------------------------------
std::cout << "constructor end" << std::endl;
}
~Listener() {
sck->close(); // A GOOD PRACTICE
// ----------------------------------------------// IF SETUP BY AN INSTANTIATION CALL INTERFACE
// aClassLocalSOCKET->close();
}
void run() {
while (1) { // recv()-messages, not interesting:
std::cout << "listening..." << std::endl;
zmq::message_t message;
sck->recv(&message);
std::cout << "received something" << std::endl;
zmq::zmq_msg_close(&message); // A GOOD PLACE TO DISCARD A NEVER MORE USED RESOURCE
}
}
};
int main(int argc, char *argv[]) {
zmq::context_t context(1); // GLOBAL CTX
Listener listener;
std::cout << "foo" << std::endl;
listener.run();
return 0;
}
资源的智能处理很重要,因为每次实例化和销毁都会在 [TIME]
域和 [SPACE]
域中承担成本(内存分配/删除分配成本,再次及时)而且这些都不便宜。
此外,应遵循 ZeroMQ Zen of Zero - 不要共享任何东西(好吧,有时共享 Context()-实例是一种方式,但是......如果你认真对待 distributed-system 设计,最好阅读一本很棒的书Pieter HINTJENS 的书“Code Connected: Volume 1”,绝对值得花时间和精力)。
关于c++ - ZMQ 导致主线程卡住(或类似的东西..?),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49141329/
当我尝试运行文件时收到此错误? can't find package zmq while executing "package require zmq" 如何在 Ubuntu 中安装 zmq? 谢谢
我已经使用 Installation on Linux / bsd without root access 中提到的步骤安装了 zmq已成功安装,但在启动应用程序时出现错误。我在没有根访问权限的环境中
我正在尝试通过 czmqz (4.0.2) 使用 libzmq (4.2.2) 在 Python 程序和我的 C++ 应用程序之间建立 IPC 通信 channel 。 我正在使用 ZMQ 发布者/订
更新我的问题 如何在我的 python zmq 服务器中表示到达的消息以显示其内容? 根据这种行为,我是否可以假设 btnState 数据无论如何都会发送到 python 服务器? 上下文: 我正在发
我有一个客户端使用一对套接字与单个服务器通信: context = zmq.Context() socket = context.socket(zmq.PAIR) socket.setsockopt(
我有一个名为 GenericMessage 的类,显示在下面的第一个代码片段中(在 GenericMessage.hxx 中定义)。 我有一个名为 TestFE.cpp 的 .cpp 文件(参见下面的
我试图在 Go 应用程序中包含 ZMQ 套接字,但 zmq4 和 gozmq(Go 的引用 ZMQ 绑定(bind)库)都给我带来了问题。我想了解为什么 zmq4 特别不能在我的系统上导入。 我运行的
我遇到了一个关于 ZeroMQ 的问题,因为我在 ZMQ 套接字上为 zmq::proxy 和 zmq::poll 使用了指针。这样做会发生错误 88 的异常(非套接字上的套接字操作)。 实际上 Ze
我正在尝试在 MQL5 中设置一个 PUB 套接字,在 Python 中设置一个 SUB 套接字来接收消息。 我在 MQL5 中有这个: #include Context context("hell
我想用 React 做一个异步路由器到经销商消息传递,但它不工作。 http://zguide.zeromq.org/php:rtdealer中的代码正在工作,但我无法确定我在做什么不同。我正在使用
我有一个 Python 脚本,我在其中绑定(bind)了多个(例如 5 个)ZMQ 接收器套接字,如下所示: receiver_1 = context.socket(zmq.PULL) receive
在Linux-Ubuntu上安装ZeroMQ,一个网站 https://tuananh.org/2015/06/16/how-to-install-zeromq-on-ubuntu/ 说要运行以下命令
如何创建允许多个发布者和这些发布者的多个订阅者的网络? 还是绝对有必要使用消息代理? import time import zmq from multiprocessing import Proces
我研究 zmq 有一段时间了,并实现了一个简化的 poc - 模仿我的基础架构设计 - 使用它(特别是使用 NetMQ 包装器),取得了很好的效果。 我的情况是这样的: 将来我计划在一台机器上运行多个
我一直在阅读 ZMQ documentation在心跳上并阅读应该使用乒乓方法而不是用于偏执海盗模式的方法 For Paranoid Pirate, we chose the second appro
我正在寻找一种支持 ZMQ 的解决方案,用于连接到 0 或 1 个对等点的绑定(bind)端点之间的通信,仅此而已。通信是双向的,连接可以随时结束或切断;并且可以与新的对等点或同一对等点重新建立连接。
我正在使用 ZMQ PUB 套接字来发布不同主题的新闻。但是其中一些消息的计算成本很高。有没有办法有一个 on_subscribe 回调,这样我就可以只计算实际需要的东西? 最佳答案 只是为了记录我找
我有兴趣了解 ZMQ 在发送消息之前是否已经压缩了消息,这样我自己就不会这样做,因为压缩消息两次是多余的。 如果确实这样做,它是自动的,还是有一个选项参数可以指定? 我正在使用java,但这确实不重要
我在我的 Java 应用程序中使用 ZMQ。我发现它的行为不均匀,即如果我发送大约 100 条消息,其中一个消费者说需要 1 秒,那么如果我们继续增加消费者,所花费的时间将变为 2,1.5,3,这样。
所以我有 8 个工作线程(PULL 套接字),它们从单个绑定(bind)的 PUSH 套接字提供数据。它们每秒处理大量数据,有时会随机崩溃。显然,我应该尝试处理这些崩溃,但我很好奇这个系统目前的弹性如
我是一名优秀的程序员,十分优秀!