zmq源码分析之消息可读通知机制

文章目录

    • 核心流程
    • 详细实现
      • [1. 网络数据接收与解析](#1. 网络数据接收与解析)
      • [2. 消息传递到 Socket](#2. 消息传递到 Socket)
      • [3. 用户层通知机制](#3. 用户层通知机制)
      • [4. zmq_poll 实现](#4. zmq_poll 实现)
    • 技术要点
      • [1. 消息完整性检测](#1. 消息完整性检测)
      • [2. 通知机制](#2. 通知机制)
      • [3. 线程安全](#3. 线程安全)
    • 完整流程图示
    • 总结

核心流程

ZeroMQ 检测完整消息可读并通知用户的完整流程

  1. 网络数据接收stream_engine 从网络读取数据
  2. 消息解析decoder 解析数据,检测完整消息
  3. 消息传递 :通过 pipe 传递给 session
  4. 消息存储socket 将消息存储在内部队列
  5. 通知用户:通过文件描述符的可读事件通知用户

详细实现

1. 网络数据接收与解析

stream_engine 接收数据

cpp 复制代码
bool zmq::stream_engine_base_t::in_event_internal ()
{
    // 读取网络数据
    const int rc = read (_inpos, bufsize);
    
    // 解码数据
    int rc = _decoder->decode (_inpos, _insize);
    
    // 消息就绪时
    if (_decoder->message_ready ()) {
        msg_t *msg = _decoder->msg ();
        // 写入 pipe
        _pipe->write (msg);
        _pipe->flush ();
    }
}

2. 消息传递到 Socket

session 处理

cpp 复制代码
void zmq::session_base_t::read_activated (zmq::pipe_t *pipe_)
{
    // 从 pipe 读取消息
    msg_t msg;
    while (pull_msg (&msg) == 0) {
        // 处理消息
        // 写入 socket 的 pipe
    }
}

socket 接收

cpp 复制代码
void zmq::socket_base_t::read_activated (zmq::pipe_t *pipe_)
{
    // 从 pipe 读取消息
    msg_t msg;
    while (pipe_->read (&msg)) {
        // 存储到内部队列
        _fq.push (&msg);
        
        // 通知用户(关键步骤)
        if (_fq.size () == 1) {
            // 第一次有消息时,通知用户
            signaler->send ();
        }
    }
}

3. 用户层通知机制

文件描述符通知

  • ZeroMQ 为每个 socket 创建一个通知文件描述符
  • 当有消息可读时,通过 signaler 向该文件描述符写入数据
  • 用户可以通过 ZMQ_FD 获取该文件描述符
  • 使用 poll/select 等系统调用监听该文件描述符

用户 API

cpp 复制代码
// 获取通知文件描述符
zmq_fd_t fd;
size_t fd_size = sizeof (fd);
zmq_getsockopt (socket, ZMQ_FD, &fd, &fd_size);

// 监听文件描述符
struct pollfd pfd;
pfd.fd = fd;
pfd.events = POLLIN;
poll (&pfd, 1, -1);

// 检查是否有消息
uint32_t events;
size_t events_size = sizeof (events);
zmq_getsockopt (socket, ZMQ_EVENTS, &events, &events_size);
if (events & ZMQ_POLLIN) {
    // 有消息可读
    zmq_recv (socket, buffer, sizeof (buffer), 0);
}

4. zmq_poll 实现

socket_poller 检测

cpp 复制代码
int zmq::socket_poller_t::check_events (zmq::socket_poller_t::event_t *events_,
                                        int n_events_)
{
    int found = 0;
    for (items_t::iterator it = _items.begin (), end = _items.end ();
         it != end && found < n_events_; ++it) {
        if (it->socket) {
            // 检查 socket 事件
            uint32_t events;
            it->socket->getsockopt (ZMQ_EVENTS, &events, &events_size);
            if (it->events & events) {
                // 有事件发生
                events_[found].socket = it->socket;
                events_[found].events = it->events & events;
                ++found;
            }
        }
    }
    return found;
}

技术要点

1. 消息完整性检测

  • decoder:负责解析网络数据,检测消息边界
  • message_ready:当解析出完整消息时返回 true
  • 帧边界:ZeroMQ 消息由多个帧组成,decoder 负责检测帧边界

2. 通知机制

  • signaler:跨平台的信号机制,用于通知用户
  • 文件描述符:每个 socket 对应一个通知文件描述符
  • 事件状态 :通过 ZMQ_EVENTS 获取 socket 的事件状态

3. 线程安全

  • 线程安全 socket:使用 signaler 机制
  • 非线程安全 socket:直接使用文件描述符
  • poller:支持同时监听多个 socket

完整流程图示

复制代码
┌─────────────────────────────────────────────────────────────────────┐
│                        消息可读通知流程                               │
├─────────────────────────────────────────────────────────────────────┤
│                                                                     │
│  1. 网络数据到达                                                   │
│     └─> stream_engine::in_event()                               │
│         └─> 读取网络数据                                          │
│         └─> decoder::decode() 解析数据                            │
│         └─> 检测消息完整性                                         │
│         └─> 完整消息就绪                                           │
│                                                                     │
│  2. 消息传递                                                       │
│     └─> _pipe->write(msg) 写入 pipe                               │
│     └─> session::read_activated() 处理                            │
│     └─> socket::read_activated() 接收                             │
│         └─> _fq.push(msg) 存储到队列                              │
│         └─> signaler->send() 发送通知                             │
│                                                                     │
│  3. 用户检测                                                       │
│     └─> poll() 检测文件描述符可读                                 │
│     └─> zmq_getsockopt(ZMQ_EVENTS) 检查事件                      │
│     └─> zmq_recv() 读取消息                                      │
│                                                                     │
└─────────────────────────────────────────────────────────────────────┘

总结

ZeroMQ 检测完整消息可读并通知用户的机制

  1. 消息解析decoder 解析网络数据,检测完整消息
  2. 消息存储:消息存储在 socket 的内部队列
  3. 通知触发 :通过 signaler 触发文件描述符的可读事件
  4. 用户检测 :用户通过 pollzmq_poll 检测事件
  5. 消息读取 :调用 zmq_recv 读取消息

这种设计确保了用户能够及时得知有消息可读,同时避免了忙等带来的性能损耗,是 ZeroMQ 高效通信的重要基础。

相关推荐
techdashen2 小时前
不开端口,不配 DNS,用树莓派在家搭一个公网可访问的 Web 服务
前端·网络·智能路由器
笨熊呆呆瓜2 小时前
【可靠性配置】华为M-LAG防环机制
网络
郑寿昌2 小时前
虚幻引擎UE5 Lumen兼容PBR材质全解析
服务器·网络·材质
ch3nyuyu2 小时前
IO缓冲区
linux·服务器
奇妙之二进制2 小时前
zmq源码分析之session
网络
奇妙之二进制2 小时前
zmq源码分析之PUSH/PULL 模式的负载均衡分析
运维·网络·负载均衡
wheeldown2 小时前
2026年4月横评三款主流远控软件实况实测:UU远程,Todesk,向日葵,综合性能 UU 远程表现最佳
linux·运维·服务器
pixcarp2 小时前
Nginx实战部署与踩坑总结 附带详细配置教程
服务器·前端·后端·nginx·golang
va学弟2 小时前
Agent入门开发(2):个性化功能添加
java·服务器·ai