Linux下使用ZMQ实践之监控事件

2024-02-02 21:38

本文主要是介绍Linux下使用ZMQ实践之监控事件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 前言

继续《Linux下使用ZMQ实践“生产者-消费者”模型》 文章之后进一步思考:
ZeroMQ通过隐藏了基础的socket操作,达到调用简明易懂的层次;
那么,如果某些场景下,又需要考虑到连接状态的维护,应该如何操作?
ZeroMQ给出的解决方案就是zmq_socket_monitor

2. 相关知识

支持监控的事件:

事件描述
ZMQ_EVENT_CONNECTEDsocket已被成功连接
ZMQ_EVENT_CONNECT_DELAYED连接动作被挂起
ZMQ_EVENT_CONNECT_RETRIED连接失败,正在重试
ZMQ_EVENT_LISTENING监听成功
ZMQ_EVENT_BIND_FAILED绑定失败
ZMQ_EVENT_ACCEPTED接受新连接
ZMQ_EVENT_ACCEPT_FAILED接受新连接失败
ZMQ_EVENT_CLOSEDsocket关闭(主动关闭)
ZMQ_EVENT_CLOSE_FAILEDsocket关闭失败
ZMQ_EVENT_DISCONNECTED连接意外关闭(被关闭)
ZMQ_EVENT_MONITOR_STOPPED监控的socket消亡

使用思路:将要监听的sock跟monitor关联,然后创建一个额外的ZMQ_PAIR,通过pair来获取sock上的事件。

3.场景举例

根据之前的“生产者-消费者”模型的一个改进:

  • 已知固定的消费者个数,如4个;
  • 生产者等待4个消费者全部启动后,才开始发送消息;
  • 生产者发送退出消息,等待消费者断开连接后才最后退出;

在之前一对多的 Push-Pull 模式下,如果没有消费者连接,则生产者数据发送会一直阻塞,但如果有至少一个连接成功,则生产者进入发送数据阶段;在改进场景中,需求所有消费者就绪后,生产者才正式开始发送数据,达到一个理想的均衡状态。

这样,我们就依赖monitor机制的实现,监听消费者的Push套件,额外增加一个监听器monitor:

#define ADDR "tcp://127.0.0.1:555"
#define MONITOR "inproc://monitor-server"...void *sock = zmq_socket(ctx, ZMQ_PUSH);void *mon  = zmq_socket(ctx, ZMQ_PAIR);...zmq_bind(sock, ADDR);zmq_socket_monitor(sock, MONITOR, ZMQ_EVENT_ALL);zmq_connect(mon, MONITOR);...

下来,我们通过monitor等待4个消费者的连接事件,成功后才发送数据;
发送数据完成后,我们通过发送“Quit”报文来通知消费者退出进程;
完整的生产者代码如下:

void test_producer(void *ctx, int times)
{int ix = 0, cnt = 0, id = 0, event = 0;char request[1024];void *sock = zmq_socket(ctx, ZMQ_PUSH);void *mon  = zmq_socket(ctx, ZMQ_PAIR);s_set_id_num(sock, id);zmq_bind(sock, ADDR);zmq_socket_monitor(sock, MONITOR, ZMQ_EVENT_ALL);zmq_connect(mon, MONITOR);LOGN("Producer %d setup\n", id);for (cnt = 0; cnt < 4;) {event = get_monitor_event(mon, NULL, NULL);if (event == ZMQ_EVENT_ACCEPTED) {LOGN("Producer accepted\n");cnt++;}}LOGN("Producer %d start\n", id);for (ix = 0; ix < times; ix++) {snprintf(request, sizeof(request), "Data-%03d-%03d", id, ix);s_send(sock, request);LOGN("Producer %d send: %s\n", id, request);usleep(100 * 1000);}for (cnt = 0; cnt < 4;) {s_send(sock, "Quit"); // 通知一个消费者,退出一个消费者event = get_monitor_event(mon, NULL, NULL);if (event == ZMQ_EVENT_DISCONNECTED) {cnt++;}}LOGN("Producer %d stop\n", id);zmq_close(sock);
}

获取监听事件的接口为,get_monitor_event,该函数从ZeroMQ帮助手册摘抄下来:

static int get_monitor_event (void *monitor, int *value, char **address)
{   // First frame in message contains event number and valuezmq_msg_t msg;zmq_msg_init (&msg);if (zmq_msg_recv (&msg, monitor, 0) == -1)return -1; // Interrupted, presumablyassert (zmq_msg_more (&msg));uint8_t *data = (uint8_t *) zmq_msg_data (&msg);uint16_t event = *(uint16_t *) (data);if (value) *value = *(uint32_t *) (data + 2);// Second frame in message contains event addresszmq_msg_init (&msg);if (zmq_msg_recv (&msg, monitor, 0) == -1)return -1; // Interrupted, presumablyassert (!zmq_msg_more (&msg));if (address) {uint8_t *data = (uint8_t *) zmq_msg_data (&msg);size_t size = zmq_msg_size (&msg); *address = (char *) malloc (size + 1);memcpy (*address, data, size);(*address)[size] = 0;}return event;
}

然后消费者的实现,跟先前的例子差不多,就多了一个退出的判断:

int test_consumer(void *ctx, int id)
{int cnt = 0;char request[1024];void *sock = zmq_socket(ctx, ZMQ_PULL);s_set_id_num(sock, id);zmq_connect(sock, ADDR);LOGN("Consumer %d start\n", id);while (++cnt) {s_recv(sock, request);LOGN("Consumer %d recv: %s\n", id, request);usleep(300 * 1000);if (strcmp(request, "Quit") == 0) {break;}}LOGN("Consumer %d stop\n", id);zmq_close(sock);
}

最后,main函数功能,主要为fork,主进程做生产者,子进程做消费者;
同时,为了方便起见,省略了waitpid回收子进程的动作;


int main(int argc, char *argv[])
{int ix = 0;void *ctx = zmq_ctx_new();srandom(time(NULL));/* 1x producter vs 4x consumer */for (ix= 1; ix <= 4; ix++) {pid_t pid = fork();if (pid == 0) {test_consumer(ctx, ix);goto out;}}test_producer(ctx, atoi(argv[1]));// TODO waitpid
out:zmq_ctx_destroy(ctx);exit(EXIT_SUCCESS);
}

实际运行情况如下:

[ 1561228921.433 ]: Consumer 1 start
[ 1561228921.433 ]: Consumer 2 start
[ 1561228921.434 ]: Consumer 4 start
[ 1561228921.434 ]: Consumer 3 start
[ 1561228921.434 ]: Producer 0 setup
[ 1561228921.435 ]: Producer accepted
[ 1561228921.496 ]: Producer accepted
[ 1561228921.572 ]: Producer accepted
[ 1561228921.572 ]: Producer accepted
[ 1561228921.572 ]: Producer 0 start
[ 1561228921.572 ]: Producer 0 send: Data-000-000
[ 1561228921.574 ]: Consumer 3 recv: Data-000-000
[ 1561228921.673 ]: Producer 0 send: Data-000-001
[ 1561228921.774 ]: Producer 0 send: Data-000-002
[ 1561228921.775 ]: Consumer 2 recv: Data-000-002
[ 1561228921.876 ]: Producer 0 send: Data-000-003
[ 1561228921.877 ]: Consumer 1 recv: Data-000-003
[ 1561228921.978 ]: Producer 0 send: Data-000-004
[ 1561228921.978 ]: Consumer 4 recv: Data-000-004
[ 1561228922.079 ]: Producer 0 send: Data-000-005
[ 1561228922.081 ]: Consumer 3 recv: Data-000-005
[ 1561228922.183 ]: Producer 0 send: Data-000-006
[ 1561228922.284 ]: Producer 0 send: Data-000-007
[ 1561228922.285 ]: Consumer 2 recv: Data-000-007
[ 1561228922.386 ]: Producer 0 send: Data-000-008
[ 1561228922.387 ]: Consumer 1 recv: Data-000-008
[ 1561228922.488 ]: Producer 0 send: Data-000-009
[ 1561228922.488 ]: Consumer 4 recv: Data-000-009
[ 1561228922.590 ]: Consumer 3 recv: Quit
[ 1561228922.892 ]: Consumer 3 stop
[ 1561228922.894 ]: Consumer 2 recv: Quit
[ 1561228923.195 ]: Consumer 2 stop
[ 1561228923.196 ]: Consumer 1 recv: Quit
[ 1561228923.497 ]: Consumer 1 stop
[ 1561228923.499 ]: Consumer 4 recv: Quit
[ 1561228923.800 ]: Consumer 4 stop
[ 1561228923.802 ]: Producer 0 stop

可以看出,程序第一阶段,启动进程;第二阶段,发送数据,负载均衡;第三阶段,回收资源。

4 结论

ZMQ监控事件的方法,提供了一种可选的扩展场景支持,实际使用可以放主线程处理,也可以放独立的子线程处理。

这篇关于Linux下使用ZMQ实践之监控事件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/672046

相关文章

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

基于Linux的ffmpeg python的关键帧抽取

《基于Linux的ffmpegpython的关键帧抽取》本文主要介绍了基于Linux的ffmpegpython的关键帧抽取,实现以按帧或时间间隔抽取关键帧,文中通过示例代码介绍的非常详细,对大家的学... 目录1.FFmpeg的环境配置1) 创建一个虚拟环境envjavascript2) ffmpeg-py

python使用库爬取m3u8文件的示例

《python使用库爬取m3u8文件的示例》本文主要介绍了python使用库爬取m3u8文件的示例,可以使用requests、m3u8、ffmpeg等库,实现获取、解析、下载视频片段并合并等步骤,具有... 目录一、准备工作二、获取m3u8文件内容三、解析m3u8文件四、下载视频片段五、合并视频片段六、错误

gitlab安装及邮箱配置和常用使用方式

《gitlab安装及邮箱配置和常用使用方式》:本文主要介绍gitlab安装及邮箱配置和常用使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装GitLab2.配置GitLab邮件服务3.GitLab的账号注册邮箱验证及其分组4.gitlab分支和标签的

springboot项目中整合高德地图的实践

《springboot项目中整合高德地图的实践》:本文主要介绍springboot项目中整合高德地图的实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一:高德开放平台的使用二:创建数据库(我是用的是mysql)三:Springboot所需的依赖(根据你的需求再

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

MySQL MCP 服务器安装配置最佳实践

《MySQLMCP服务器安装配置最佳实践》本文介绍MySQLMCP服务器的安装配置方法,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下... 目录mysql MCP 服务器安装配置指南简介功能特点安装方法数据库配置使用MCP Inspector进行调试开发指

nginx启动命令和默认配置文件的使用

《nginx启动命令和默认配置文件的使用》:本文主要介绍nginx启动命令和默认配置文件的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录常见命令nginx.conf配置文件location匹配规则图片服务器总结常见命令# 默认配置文件启动./nginx

在Windows上使用qemu安装ubuntu24.04服务器的详细指南

《在Windows上使用qemu安装ubuntu24.04服务器的详细指南》本文介绍了在Windows上使用QEMU安装Ubuntu24.04的全流程:安装QEMU、准备ISO镜像、创建虚拟磁盘、配置... 目录1. 安装QEMU环境2. 准备Ubuntu 24.04镜像3. 启动QEMU安装Ubuntu4

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解