ypipe_t has a yqueue_t. pipe_t relates two ypipe(s)。pipe_t就是0MQ框架内使用的底层队列。

yqueue_t的设计目的。

yqueue_t 的结构成员

这个队列的设计与std::deque有的相似,都用chunk双向链表实现,同样避免FIFO队列对内存过度频繁的分配和释放。yqueue_t不支持随机访问,以及遍历迭代。yqueue_t采用了一个spare_chunk,保存最近一次访问过后释放或分配出来的chunk,提高空闲chunk的cache亲和性。并且spare_chunk使用了lock-free的atomic_ptr_t。

begin_chunk指向chunk链表的开始端,back_chunk指向chunk链表的有效尾端。

yqueue只支持类似std::deque的pop_front, pop_back, push_back。

pop_front当消耗完一个chunk,然后空闲出一个chunk,替换spare_chunk。

push_back当接近可写chunk的末端,就会分配出一个chunk或取出spare_chunk,链入到链表尾端,end_chunk指向这个新的chunk。

pop_back可能会使当前可写chunk空闲,但这个空闲出来的chunk不会放到spare_chunk。

接着就是ypipe_t, 其设计目的:

ypipe_t 结构成员:

这里的lock-free,同样关键在atomic_ptr_t类型的成员。同样是利用cas(compare_and_swap)替代解决某些情况的锁。这里并没有实现RCU这类lock-free读写并发算法,也不支持读之间的并发。设计目的,明确指出,只有一条线程可以在任何指定时刻读,以及一条线程可以在任何指定时刻写,换句话说,读写可以无锁并发,但读之间不可并发,写之间也不可以并发。

这里的读写lock-free,w指向yqueue的back,r指向yqueue的front,f指向yqueue未flush的位置。也就是f在w前,r是不会指向到f和w之间的位置,也就是yqueue未空也是不可读的。读写之前的竞态关键就是当r到达了f的位置。这时使用了一个atomic_ptr_t类型的c去进行lock-free解决。准确来说,竞态可能会发生在读线程的读和写线程的flush时刻。

如果你要使用这个ypipe作为你的程序的队列的话,你还是得用同步原语控制读之间以及写之间的同步。最理想的使用就是,同时只有一条线程可读,另一条线程可写,读写两条线程之间是可以lock-free并发的。如果用锁的话,当发生竞态就有一条线程要进入内核态等待,另一条线程也要进入内核态去唤醒。详细请看前面关于futex的文章。

最后就是pipe_t。

主要结构成员:

pipe_t关联inpipe的读端,关联outpipe的写端。在这一层并提供一个事件槽。

==================================================================

ypipe_t用于两个小组件, 一个是IO消息的底层upipe_t,上面封装一层pipe_t,用于某些类型的0MQ socket (socket_base_t),另一个是cmd命令的cpipe_t用于mailbox,而mailbox是向io_thread_t以及rearer_t对象通讯的队列。mailbox并不是底层io文件,但借用了signaler_t(实现用某种文件系统文件),并实现i_poll_event接口,让mailbox的"io"纳入到操作系统的poller。当通过mailbox发送命令与mailbox的拥有者通讯时,就会在某个reactor线程中处理事件。参看mailbox::send

==========================================================

整个zmq框架内部,对象之间使用mailbox进行通讯。

来看一下zmq context

cxt_t将reaper_t回收线程,io_thread_t 以及 socket_base_t 的mailbox 按放在 slots 指针数组。

slots[0] = &term_mailbox

slots[1] = reaper->get_mailbox()

slots[2 .. 2+io_nr] = io_thread->get_mailbox()

slots[...] = socket_base->get_mailbox()

这里要清楚,

1. 可以向一个io_thread发送command,并让command在目标io_thread上运行。

2. 向socket_base_t发送command,不会唤醒io_thread,因为socket_base_t不负责io,负责io的stream_engine_t,这才是我们通常理解的stream connection。你zmq_poll这个socket_base_t的控制线程会被唤醒,并执行这个命令。但是一般不会直接向socket_base_t发送command。而是通常继承它的tid(实质是slots的索引)的pipe_t,在写入pipe_t后,向pipe_t发送send_activate_read(send_command),就会最终向它的父对象(不是继承关系,而是创建关系)socket_base_t发送command。

======================================================

在zmq框架内部有这样一张 pipe 网络。

1. 首先0MQ socket 不是我们理解的传统意义的socket(某个连接的插口对),一个0MQ socket可以有多个连接,并且是透明的。

2. 每个0MQ socket会一对 pipe 用于编程者的访问。

3. 0MQ socket里面每个连接(session)都有一对 pipe,0MQ socket会透明地将消息在它的 pipe 和 多个session的pipe之间进行交换。

4. ctx_t包含了slots(mailbox,一种借用了signaler可以被socket poll的pipe),对象间的消息通讯使用这些mailbox。

5. zmq_proxy,透明地将一个socket_base_t的pipe和另一个socket_base_t的pipe的消息进行交换。

0MQ底层队列设计的更多相关文章

  1. 基于redis的延迟消息队列设计

    需求背景 用户下订单成功之后隔20分钟给用户发送上门服务通知短信 订单完成一个小时之后通知用户对上门服务进行评价 业务执行失败之后隔10分钟重试一次 类似的场景比较多 简单的处理方式就是使用定时任务 ...

  2. 基于redis的延迟消息队列设计(转)

    需求背景 用户下订单成功之后隔20分钟给用户发送上门服务通知短信 订单完成一个小时之后通知用户对上门服务进行评价 业务执行失败之后隔10分钟重试一次 类似的场景比较多 简单的处理方式就是使用定时任务 ...

  3. ActiveMQ学习总结(8)——消息队列设计精要

    消息队列已经逐渐成为企业IT系统内部通信的核心手段.它具有低耦合.可靠投递.广播.流量控制.最终一致性等一系列功能,成为异步RPC的主要手段之一. 当今市面上有很多主流的消息中间件,如老牌的Activ ...

  4. Apache Druid 底层存储设计(列存储与全文检索)

    导读:首先你将通过这篇文章了解到 Apache Druid 底层的数据存储方式.其次将知道为什么 Apache Druid 兼具数据仓库,全文检索和时间序列的特点.最后将学习到一种优雅的底层数据文件结 ...

  5. 算法与数据结构题目的 PHP 实现:栈和队列 设计一个有 getMin 功能的栈

    刚入手了一本<程序员代码面试指南>,书中题目的代码都是 Java 实现的,琢磨着把这些代码用 PHP 敲一遍,加深印象. 题目:设计一个有 getMin 功能的栈 —— 实现一个特殊的栈, ...

  6. Python 番外 消息队列设计精要

    消息队列已经逐渐成为企业IT系统内部通信的核心手段.它具有低耦合.可靠投递.广播.流量控制.最终一致性等一系列功能,成为异步RPC的主要手段之一.当今市面上有很多主流的消息中间件,如老牌的Active ...

  7. Asp.Net Core 快速邮件队列设计与实现

    发送邮件几乎是软件系统中必不可少的功能,在Asp.Net Core 中我们可以使用MailKit发送邮件,MailKit发送邮件比较简单,网上有许多可以参考的文章,但是应该注意附件名长度,和附件名不能 ...

  8. C中级 消息队列设计

    引言  - 补充好开始 消息队列在游戏服务器层应用非常广泛. 应用于各种耗时的IO操作业务上.消息队列可以简单理解为 [消息队列 = 队列 + 线程安全]本文参照思路如下, 最后献上一个大神们斗法的场 ...

  9. 【Java TCP/IP Socket】深入剖析socket——TCP通信中由于底层队列填满而造成的死锁问题(含代码)

    基础准备 首先需要明白数据传输的底层实现机制,在http://blog.csdn.net/ns_code/article/details/15813809这篇博客中有详细的介绍,在上面的博客中,我们提 ...

随机推荐

  1. 并发编程之Fork/Join

    并发与并行 并发:多个进程交替执行. 并行:多个进程同时进行,不存在线程的上下文切换. 并发与并行的目的都是使CPU的利用率达到最大.Fork/Join就是为了尽可能提高硬件的使用率而应运而生的. 计 ...

  2. 利用ansible书写playbook在华为云上批量配置管理工具自动化安装ceph集群

    首先在华为云上购买搭建ceph集群所需云主机: 然后购买ceph所需存储磁盘 将购买的磁盘挂载到用来搭建ceph的云主机上 在跳板机上安装ansible 查看ansible版本,检验ansible是否 ...

  3. GO实现简单(命令行)工具:sftp,文檔压解,RDS备份,RDS备份下载

    GO实现简单(命令行)工具:sftp,文檔压解,RDS备份,RDS备份下载 轉載請註明出處:https://www.cnblogs.com/funnyzpc/p/11721978.html 内容提要: ...

  4. 百万年薪python之路 -- 基本数据类型

    整数 -- 数字(int) 用于比较和运算 32位 2 ** 31 ~ 2 ** 31-1 64位 -2 ** 63 ~ 2 ** 63- 1 ​ + - * / // ** % python2 整型 ...

  5. lamda和匿名内部类

    匿名内部类 匿名内部类在日常编程中还是经常会使用的.比如 ArrayList<String> list=new ArrayList<>(); list.add(new Stri ...

  6. SQL注入学习(一)

    SQL注入(SQL Injection)是一种常见的Web安全漏洞,攻击者利用这个漏洞,可以访问或修改数据,或者利用潜在的数据库漏洞进行攻击. SQL注入基础 漏洞原理 针对SQL注入的攻击行为可描述 ...

  7. 通过FeignClient接收shaded的javabean的JSON序列化

    问题说明 最近做了关于flink的需求. 现在需要通过HTTP访问FLINK的 RESTAPI, rest 接口的JSON 非常庞大而复杂. 那么怎么去完整的接收数据呢? 方法一就是手写部分需要的Ja ...

  8. django-URL之include函数(五)

    三种格式:(1)incude(module,namespace=None) from django.urls import path,include from book import urls url ...

  9. SpringBoot与MybatisPlus3.X整合之通用枚举(十二)

    一 通用枚举 解决了繁琐的配置,让 mybatis 优雅的使用枚举属性! 自3.1.0开始,可配置默认枚举处理类来省略扫描通用枚举配置 默认枚举配置 升级说明: 3.1.0 以下版本改变了原生默认行为 ...

  10. Linux进程组和会话

    Linux的进程相互之间有一定的关系.比如说,在Linux进程基础中,我们看到,每个进程都有父进程,而所有的进程以init进程为根,形成一个树状结构.我们在这里讲解进程组和会话,以便以更加丰富的方式了 ...