当前位置: 首页 > article >正文

zmq源码分析之io_thread_t

文章目录概述继承关系核心成员构造函数启动与停止启动停止事件处理读事件处理核心其他事件理论上不会被调用停止处理架构图事件循环流程与其他组件的关系线程创建流程关键设计点命令处理类型性能特点总结概述io_thread_t是 ZeroMQ 中的I/O线程负责处理网络事件和命令传递。它是 ZeroMQ 多线程架构的核心组件运行独立的事件循环来监听和处理各种文件描述符上的I/O事件。io_thread_t通过poller_t间接的持有线程同时io_thread_t自身也持有了一个邮箱用于接收外部命令邮箱本身也是个文件描述符io_thread_t自身的io事件也交给了poller_t管理。继承关系classio_thread_tZMQ_FINAL:publicobject_t,publici_poll_eventsobject_t提供命令处理能力和对象标识i_poll_events提供I/O事件回调接口核心成员private:// I/O线程通过这个邮箱访问传入的命令mailbox_t _mailbox;// 与邮箱文件描述符关联的句柄poller_t::handle_t _mailbox_handle;// I/O多路复用使用poller对象实现poller_t*_poller;构造函数zmq::io_thread_t::io_thread_t(ctx_t*ctx_,uint32_ttid_):object_t(ctx_,tid_),_mailbox_handle(static_castpoller_t::handle_t(NULL)){// 创建poller对象_pollernew(std::nothrow)poller_t(*ctx_);alloc_assert(_poller);// 将邮箱的文件描述符注册到pollerif(_mailbox.get_fd()!retired_fd){_mailbox_handle_poller-add_fd(_mailbox.get_fd(),this);_poller-set_pollin(_mailbox_handle);}}关键操作创建平台特定的pollerepoll/kqueue/poll/select将邮箱的文件描述符注册到poller设置监听读事件POLLIN启动与停止启动voidzmq::io_thread_t::start(){charname[16];snprintf(name,sizeof(name),IO/%u,get_tid()-zmq::ctx_t::reaper_tid-1);// 启动底层的I/O线程_poller-start(name);}停止voidzmq::io_thread_t::stop(){send_stop();}事件处理读事件处理核心voidzmq::io_thread_t::in_event(){// 从邮箱接收命令并处理command_t cmd;intrc_mailbox.recv(cmd,0);while(rc0||errnoEINTR){if(rc0)cmd.destination-process_command(cmd);rc_mailbox.recv(cmd,0);}errno_assert(rc!0errnoEAGAIN);}工作流程从邮箱接收命令循环处理所有可用命令直到邮箱为空返回 EAGAIN其他事件理论上不会被调用voidzmq::io_thread_t::out_event(){// 永远不会在这里轮询POLLOUTzmq_assert(false);}voidzmq::io_thread_t::timer_event(int){// 这里没有定时器永远不会被调用zmq_assert(false);}停止处理voidzmq::io_thread_t::process_stop(){zmq_assert(_mailbox_handle);_poller-rm_fd(_mailbox_handle);_poller-stop();}架构图┌─────────────────────────────────────────────────────────────────────┐ │ io_thread_t │ ├─────────────────────────────────────────────────────────────────────┤ │ │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ object_t (基类) │ │ │ │ - ctx_t * 上下文指针 │ │ │ │ - uint32_t tid 线程ID │ │ │ │ - process_command() 命令处理分发 │ │ │ └─────────────────────────────────────────────────────────────┘ │ │ │ │ │ ▼ │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ i_poll_events (接口) │ │ │ │ in_event() 文件描述符可读回调 │ │ │ │ out_event() 文件描述符可写回调 │ │ │ │ timer_event() 定时器到期回调 │ │ │ └─────────────────────────────────────────────────────────────┘ │ │ │ │ │ ▼ │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ ┌─────────────┐ ┌─────────────┐ ┌─────────────┐ │ │ │ │ │ mailbox_t │ │ poller_t │ │mailbox_handle│ │ │ │ │ │ (命令队列) │ │ (事件循环) │ │ (fd句柄) │ │ │ │ │ └──────┬──────┘ └──────┬──────┘ └─────────────┘ │ │ │ │ │ │ │ │ │ └─────────┼──────────────────┼───────────────────────────────┘ │ │ │ │ │ │ ▼ ▼ │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ 物理线程 (Poller 事件循环) │ │ │ │ ┌───────────────────────────────────────────────────────┐ │ │ │ │ │ poll/epoll/kqueue/select │ │ │ │ │ │ │ │ │ │ │ │ │ └── in_event() ── 处理命令 ── process_command() │ │ │ │ │ └───────────────────────────────────────────────────────┘ │ │ │ └─────────────────────────────────────────────────────────────┘ │ │ │ └─────────────────────────────────────────────────────────────────────┘事件循环流程┌─────────────────────────────────────────────────────────────────────┐ │ IO 线程事件循环 │ ├─────────────────────────────────────────────────────────────────────┤ │ │ │ ┌───────────────┐ │ │ │ Poller 等待 │ │ │ │ 事件 │ │ │ └───────┬───────┘ │ │ │ │ │ │ 邮箱fd可读 │ │ ▼ │ │ ┌───────────────┐ │ │ │ 调用in_event() │ │ │ └───────┬───────┘ │ │ │ │ │ ▼ │ │ ┌───────────────┐ ┌───────────────┐ │ │ │ mailbox.recv()│────│ 有命令 │──── 处理命令 │ │ └───────┬───────┘ └───────────────┘ │ │ │ │ │ │ EAGAIN (无命令) │ │ ▼ │ │ ┌───────────────┐ │ │ │ 返回继续等待 │ │ │ └───────────────┘ │ │ │ └─────────────────────────────────────────────────────────────────────┘与其他组件的关系┌─────────────────────────────────────────────────────────────────────┐ │ ctx_t (上下文) │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ _io_threads: IO线程池 │ │ │ │ ┌────────┐ ┌────────┐ ┌────────┐ │ │ │ │ │IO[0] │ │IO[1] │ │IO[2] │ ... │ │ │ │ └───────┘ └───────┘ └───────┘ │ │ │ │ │ │ │ │ │ │ └───────┼─────────┼─────────┼─────────────────────────────────┘ │ │ │ │ │ │ │ ▼ ▼ ▼ │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ Slot 数组 │ │ │ │ [term] [reaper] [IO[0]] [IO[1]] [IO[2]] [socket1] ... │ │ │ └─────────────────────────────────────────────────────────────┘ │ └─────────────────────────────────────────────────────────────────────┘ ┌─────────────────────────────────────────────────────────────────────┐ │ Session/Engine │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ 网络连接 (TCP/UDP/IPC/PGM) │ │ │ │ │ │ │ │ │ │ 注册到 IO 线程的 Poller │ │ │ │ ▼ │ │ │ │ ┌─────────────────────────────────────────────────────┐ │ │ │ │ │ 当网络数据可读/可写时Poller 回调 io_thread_t │ │ │ │ │ │ → in_event() / out_event() │ │ │ │ │ │ → 处理网络数据 │ │ │ │ │ └─────────────────────────────────────────────────────┘ │ │ │ └─────────────────────────────────────────────────────────────┘ │ └─────────────────────────────────────────────────────────────────────┘ ┌─────────────────────────────────────────────────────────────────────┐ │ Socket (应用线程) │ │ ┌─────────────────────────────────────────────────────────────┐ │ │ │ 用户 API 调用 (send/recv/connect/bind) │ │ │ │ │ │ │ │ │ │ 发送命令到 IO 线程的 mailbox │ │ │ │ ▼ │ │ │ │ ┌─────────────────────────────────────────────────────┐ │ │ │ │ │ mailbox.send(command) │ │ │ │ │ │ → _signaler.send() (如果需要) │ │ │ │ │ └─────────────────────────────────────────────────────┘ │ │ │ └─────────────────────────────────────────────────────────────┘ │ └─────────────────────────────────────────────────────────────────────┘线程创建流程上下文创建时创建 IO 线程数组_io_threads为每个 IO 线程分配 slotIO 线程构造创建 mailbox创建 poller注册 mailbox fd 到 pollerIO 线程启动调用start()启动 pollerPoller 创建物理线程线程运行事件循环关键设计点设计点说明独立线程每个 IO 线程运行在独立物理线程上事件驱动基于 poll/epoll/kqueue/select命令处理通过 mailbox 接收命令多路复用一个 poller 监听多个 fd负载均衡ctx 选择负载最小的 IO 线程命令处理类型IO 线程可以处理多种命令stop- 停止 IO 线程plug- 注册到 IO 线程bind- 绑定 pipe 到 socketactivate_read/write- pipe 流量控制hiccup- pipe 中断处理等等…性能特点高效事件处理基于内核级 I/O 多路复用低延迟事件驱动无轮询开销可扩展支持多个 IO 线程线程安全每个 IO 线程独立运行总结io_thread_t是 ZeroMQ 多线程架构的核心组件独立运行每个 IO 线程运行在独立物理线程上事件驱动通过 poller 监听文件描述符命令通道通过 mailbox 接收来自其他线程的命令网络处理负责处理所有网络 I/O 事件负载均衡ctx 选择最适合的 IO 线程处理连接这是 ZeroMQ 能够高效处理高并发连接的关键机制

相关文章:

zmq源码分析之io_thread_t

文章目录概述继承关系核心成员构造函数启动与停止启动停止事件处理读事件处理(核心)其他事件(理论上不会被调用)停止处理架构图事件循环流程与其他组件的关系线程创建流程关键设计点命令处理类型性能特点总结概述 io_thread_t 是…...

Windows 10系统清理终极指南:让旧电脑重获新生的免费神器

Windows 10系统清理终极指南:让旧电脑重获新生的免费神器 【免费下载链接】Win10BloatRemover Configurable CLI tool to easily and aggressively debloat and tweak Windows 10 by removing preinstalled UWP apps, services and more. Originally based on the W…...

10年老兵带你学Java(第3课):数组和方法 - 代码的复用

本课目标 数组:一组数据的容器方法:代码的复用面向对象入门:类和方法的关系 上节课学了变量,一个变量存一个数据。 这节课学数组,一个变量存一组数据。还有方法,把代码打包成可复用的块。一、数组&#xff…...

Polkadot 技术栈地图 2026

作者:PokerMoon 团队 区块链项目的官网有一个通病——首页永远写得像科幻小说,“Tech” 页面永远写得像论文目录。Polkadot 的 /tech 页就是典型案例。你点进去,映入眼帘的是一连串大写字母缩写:JAM、PVM、Coretime、XCM、PoP………...

C语言学习日志

大家好,我是一名物联网工程专业的大学生,很高兴认识大家,同时这也是我的第一篇博客。1.我的编程目标: 学习c语言主要是为了对后面c的学习做铺垫,完善我的大创项目,为我的简历增添色彩,收到一份好…...

5分钟实现Windows和Linux鼠标指针美化:macOS风格光标主题完整指南

5分钟实现Windows和Linux鼠标指针美化:macOS风格光标主题完整指南 【免费下载链接】apple_cursor Free & Open source macOS Cursors. 项目地址: https://gitcode.com/gh_mirrors/ap/apple_cursor 你是否厌倦了Windows或Linux系统默认的单调鼠标指针&…...

计算机毕业设计:Python农产品价格与销量关联分析系统 Django框架 数据分析 可视化 大数据 大模型 机器学习(建议收藏)✅

1、项目介绍 技术栈 采用 Python 语言开发,基于 Django 框架搭建后端服务,使用 MySQL 数据库进行数据存储,前端结合 HTML 与 Echarts 可视化库实现数据展示。 功能模块农产品价格区间分布农产品销量分布农产品不同省份销量分布农产品…...

Cats Blender插件终极指南:如何快速将3D模型优化并导入VRChat

Cats Blender插件终极指南:如何快速将3D模型优化并导入VRChat 【免费下载链接】cats-blender-plugin :smiley_cat: A tool designed to shorten steps needed to import and optimize models into VRChat. Compatible models are: MMD, XNALara, Mixamo, DAZ/Poser,…...

阿贝云免费服务器

阿贝云免费虚拟主机https://www.abeiyun.com,点击进入,登录即领...

计算机毕业设计:Python农业电商销售数据分析平台 Django框架 数据分析 可视化 大数据 大模型 机器学习(建议收藏)✅

博主介绍:✌全网粉丝50W,前互联网大厂软件研发、集结硕博英豪成立工作室。专注于计算机相关专业项目实战8年之久,选择我们就是选择放心、选择安心毕业✌ > 🍅想要获取完整文章或者源码,或者代做,拉到文章底部即可与…...

【AGI蛋白质折叠预测革命】:2024年AlphaFold 3与RoseTTAFold AI实测对比,精准度突破99.2%的5大临床应用落地路径

第一章:AGI蛋白质折叠预测能力的范式跃迁 2026奇点智能技术大会(https://ml-summit.org) 传统蛋白质结构预测依赖于多序列比对(MSA)与共进化信号挖掘,计算密集且对低同源性蛋白失效;而新一代AGI驱动的折叠模型已突破…...

别再学框架了!2026奇点大会证实:未来3年高薪岗位只筛选这7种AGI协同行为模式

第一章:2026奇点智能技术大会:AGI与编程能力 2026奇点智能技术大会(https://ml-summit.org) AGI驱动的实时代码生成范式 本届大会首次公开展示了基于多模态具身推理的AGI编程代理——SingularityCoder v3.2。该系统不再依赖静态训练数据,而…...

ESP32上传图片到巴法云,除了HTTPClient,你还可以试试这个库

ESP32进阶方案:三种高效上传图片到巴法云的库对比与实践 在物联网项目中,ESP32作为一款高性价比的Wi-Fi/蓝牙双模芯片,经常被用于图像采集与传输场景。巴法云作为国内流行的物联网平台,提供了便捷的图片上传接口。虽然官方例程通常…...

企业云盘选型标准合同条款:数据归属/服务等级/SLA全解析

作者:巴别鸟技术团队 适用场景:IT采购、合规审查、法务评估 更新时间:2026-04引言:为什么选云盘先看合同? 企业选择云盘时,大多数人盯着功能对比、UI体验、存储价格——但真正踩过坑的IT负责人知道&#xf…...

企业云盘数据安全实战:从传输加密到存储隔离的完整方案

⚠️ 一个让人震惊的事实:根据OWASP 2023年报告,超过60%的企业数据泄露事件,攻击入口不是高深的黑客技术,而是未加密的传输层。你的企业云盘HTTP流量,可能正在被监听,而你毫不知情。 前言:一个价…...

Go语言怎么做地理围栏_Go语言地理位置计算教程【指南】

最稳的IP地理围栏方案是用oschwald/maxminddb-golang读取GeoLite2-City.mmdb,配合CF-Connecting-IP或Nginx可信IP头获取真实IP,白名单用map、围栏用S2库做点面判断,高实时场景选Tile38。用 maxminddb 查 IP 归属做围栏,别碰 geoip…...

如何备份大量小表组成的数据库_并行导出与多文件并发写入.txt

PHP开发无需选机箱,真正关键的是CPU单核性能、RAM容量和SSD读写延迟;生产服务器的硬件选型属于运维范畴,与PHP编码、调试、本地运行无关。PHP 是运行在服务器端的脚本语言,源码开发阶段根本不需要考虑机箱、散热或 PCIe 插槽——这…...

AGI不再依赖云巨头?深度拆解Polkadot+LLM+Verifiable Computation三栈融合架构(含GitHub Star 3.2K实证项目)

第一章:AGI的分布式与去中心化探索 2026奇点智能技术大会(https://ml-summit.org) 传统AGI研发范式高度依赖中心化算力集群、统一模型权重分发与集中式数据治理,而分布式与去中心化架构正催生新一代AGI演进路径:节点自治、共识驱动、价值可验…...

MySQL升级如何回滚到旧版本_灾难恢复方案与快照备份恢复.txt

text-rendering: optimizeSpeed 几乎没人用,因为现代浏览器(Chrome、Firefox、Safari)均不真正支持它,实际等同于 auto,无法实现字形简化或性能提升。text-rendering: optimizeSpeed 为什么几乎没人用因为现代浏览器基…...

UAF hacknote

hacknotep process(./hacknote, env{GLIBC_TUNABLES: glibc.malloc.tcache_count0})禁止tcache分析一下可以得到:查看del_note函数,这里只是free了两个堆,但并没有把*(&notelist)置空,即bss段上的指针依旧指向堆的地址&#…...

三步解锁Switch潜能:大气层系统从零到精通的实践指南

三步解锁Switch潜能:大气层系统从零到精通的实践指南 【免费下载链接】Atmosphere-stable 大气层整合包系统稳定版 项目地址: https://gitcode.com/gh_mirrors/at/Atmosphere-stable 大气层系统(Atmosphere)作为Nintendo Switch最稳定…...

从一道BUUCTF的SSRF题,聊聊Linux命令行那些“意想不到”的利用姿势(HITCON 2017实战复盘)

从BUUCTF SSRF题看Linux命令行的隐秘攻击面 在CTF竞赛和实际渗透测试中,SSRF(服务器端请求伪造)常被视作简单的内网探测工具,但2017年HITCON这道题却展示了它如何与Linux命令行特性结合,实现从信息泄露到远程代码执行的…...

告别VMware!用Arsenal Image Mounter在Windows里直接‘打开’取证镜像,像本地硬盘一样操作

数字取证新范式:Arsenal Image Mounter实战指南 在数字取证和安全分析领域,效率往往意味着成功的关键。想象一下这样的场景:你刚刚获取了一个关键的磁盘镜像文件,可能是.raw、.dd或.E01格式,里面可能包含着破解案件的…...

H3C交换机上给不同VLAN配DHCP,一次搞定网关、地址池和DNS(附完整命令)

H3C交换机多VLAN场景下的DHCP服务配置实战指南 在中小型企业网络或实验室环境中,随着设备数量的增加和网络结构的复杂化,手动为每台终端配置静态IP地址变得越来越不现实。这时候,DHCP(动态主机配置协议)服务就显得尤为…...

mysql如何优化索引以减少扫描_mysql高效索引设计原则

MySQL索引失效主因是最左前缀原则被破坏:范围查询或跳过中间列会导致右侧列无法使用索引;ORDER BY需满足最左连续列且排序方向一致;索引过多拖慢写入,应评估选择性与实际使用率;EXPLAIN中key_len和Extra比type更能反映…...

RKMEDIA VO图层配置与双屏显示实战

1. RKMEDIA VO模块基础解析 第一次接触RV1126/RV1109的双屏显示功能时,我也被各种专业术语搞得一头雾水。经过几个项目的实战,现在终于能把这个功能讲得通俗易懂了。RKMEDIA的VO(Video Output)模块本质上是对Linux DRM&#xff08…...

赛元SC95F8617触摸库实战:从电机干扰到人体检测,我的按摩椅项目避坑实录

赛元SC95F8617触摸库实战:从电机干扰到人体检测,我的按摩椅项目避坑实录 按摩椅作为智能家居领域的热门产品,人体检测功能的可靠性直接影响用户体验。去年接手的一个高端按摩椅项目,让我深刻体会到赛元SC95F8617触摸库在复杂电磁环…...

FPGA设计里选乘法器IP还是写RTL?从面积、时序和易用性帮你决策

FPGA乘法器设计决策指南:IP核与RTL方案的深度权衡 在数字信号处理、通信系统和图像处理等FPGA应用领域,乘法器作为基础运算单元,其实现方式直接影响着系统性能、资源占用和开发效率。面对Xilinx/Altera提供的成熟IP核与自己编写RTL代码这两种…...

从寄存器手册到代码:手把手教你逆向分析ES8311官方驱动配置逻辑

逆向工程实战:解码ES8311音频芯片寄存器配置的艺术 在嵌入式音频开发领域,真正的高手往往不是那些只会调用API的开发者,而是能够深入芯片寄存器层面,理解每一个配置位意义的"芯片侦探"。ES8311作为一款广泛应用于智能设…...

为什么92%的AGI项目在记忆对齐阶段失败?——2026奇点大会实测数据揭示5大认知断层与3步修复协议(含开源Memory-LLM v0.9预览版)

第一章:2026奇点智能技术大会:AGI与记忆系统 2026奇点智能技术大会(https://ml-summit.org) 本届大会首次将“记忆系统”确立为AGI架构的核心支柱,而非传统意义上的辅助模块。研究者指出,具备可演化、可检索、可因果回溯的长期记…...