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

C++中间件DDS介绍

C++ DDS 库简介

DDS(Data Distribution Service) 是一种用于实时分布式系统通信的中间件标准,由 OMG(Object Management Group) 提出。它是一种发布/订阅(Publish/Subscribe)模式的数据通信框架,广泛应用于嵌入式系统、物联网、航空航天、机器人等领域。

C++ DDS 库是实现 DDS 标准的库,常见的实现包括:

  • RTI Connext DDS(Real-Time Innovations 提供)
  • OpenDDS(开源实现)
  • Eclipse Cyclone DDS(开源实现)
  • Fast DDS (以前称为 FastRTPS)(由 eProsima 开发)

这些库提供了丰富的 API,用于创建发布者、订阅者、定义数据类型和 QoS(Quality of Service)等功能。


DDS 的核心概念

  1. DomainParticipant:表示 DDS 系统的入口点,属于特定的域。
  2. Publisher/Subscriber:分别用于发布和订阅数据。
  3. Topic:数据通信的主题,定义通信数据的类型。
  4. DataWriter/DataReader:负责数据的写入与读取。
  5. QoS:定义数据传输的质量服务属性。

一个简单的 C++ DDS 代码示例

以下是使用 RTI Connext DDS 的一个简单例子,展示如何发布和订阅一个简单的消息数据类型。请确保你已安装 RTI Connext DDS 或其他 DDS 实现,并正确配置开发环境。

1. 定义数据类型(IDL 文件)
module HelloWorld {struct Msg {long id;string message;};
};

IDL (Interface Definition Language) 文件用于定义消息的数据结构,DDS 会通过工具生成相应的 C++ 类型。

2. 发布者(Publisher)代码
#include <iostream>
#include "HelloWorld/Msg.hpp" // 自动生成的头文件
#include "ndds/ndds_cpp.h"   // RTI Connext DDS 的核心头文件int main() {// 1. 创建 DomainParticipantDDSDomainParticipant *participant = DDSTheParticipantFactory->create_participant(0, DDS_PARTICIPANT_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (participant == nullptr) {std::cerr << "创建 DomainParticipant 失败!" << std::endl;return -1;}// 2. 注册数据类型const char *type_name = HelloWorld::MsgTypeSupport::get_type_name();if (HelloWorld::MsgTypeSupport::register_type(participant, type_name) != DDS_RETCODE_OK) {std::cerr << "注册数据类型失败!" << std::endl;return -1;}// 3. 创建 TopicDDSTopic *topic = participant->create_topic("HelloWorldTopic", type_name, DDS_TOPIC_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (topic == nullptr) {std::cerr << "创建 Topic 失败!" << std::endl;return -1;}// 4. 创建 Publisher 和 DataWriterDDSPublisher *publisher = participant->create_publisher(DDS_PUBLISHER_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (publisher == nullptr) {std::cerr << "创建 Publisher 失败!" << std::endl;return -1;}DDSDataWriter *writer = publisher->create_datawriter(topic, DDS_DATAWRITER_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (writer == nullptr) {std::cerr << "创建 DataWriter 失败!" << std::endl;return -1;}HelloWorld::MsgDataWriter *msg_writer = HelloWorld::MsgDataWriter::narrow(writer);if (msg_writer == nullptr) {std::cerr << "narrow DataWriter 失败!" << std::endl;return -1;}// 5. 发布数据HelloWorld::Msg msg;msg.id = 1;msg.message = DDS_String_dup("Hello, DDS!");if (msg_writer->write(msg, DDS_HANDLE_NIL) != DDS_RETCODE_OK) {std::cerr << "写入数据失败!" << std::endl;} else {std::cout << "消息已发布: " << msg.message << std::endl;}// 清理资源participant->delete_contained_entities();DDSTheParticipantFactory->delete_participant(participant);return 0;
}

3. 订阅者(Subscriber)代码
#include <iostream>
#include "HelloWorld/Msg.hpp"
#include "ndds/ndds_cpp.h"class MsgListener : public DDSDataReaderListener {
public:void on_data_available(DDSDataReader *reader) override {HelloWorld::MsgDataReader *msg_reader = HelloWorld::MsgDataReader::narrow(reader);if (msg_reader == nullptr) return;HelloWorld::MsgSeq data_seq;DDS_SampleInfoSeq info_seq;DDS_ReturnCode_t retcode = msg_reader->take(data_seq, info_seq, DDS_LENGTH_UNLIMITED,DDS_ANY_SAMPLE_STATE, DDS_ANY_VIEW_STATE, DDS_ANY_INSTANCE_STATE);if (retcode == DDS_RETCODE_OK) {for (int i = 0; i < data_seq.length(); ++i) {if (info_seq[i].valid_data) {std::cout << "接收到消息: id=" << data_seq[i].id<< ", message=" << data_seq[i].message << std::endl;}}msg_reader->return_loan(data_seq, info_seq);}}
};int main() {DDSDomainParticipant *participant = DDSTheParticipantFactory->create_participant(0, DDS_PARTICIPANT_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (participant == nullptr) {std::cerr << "创建 DomainParticipant 失败!" << std::endl;return -1;}const char *type_name = HelloWorld::MsgTypeSupport::get_type_name();if (HelloWorld::MsgTypeSupport::register_type(participant, type_name) != DDS_RETCODE_OK) {std::cerr << "注册数据类型失败!" << std::endl;return -1;}DDSTopic *topic = participant->create_topic("HelloWorldTopic", type_name, DDS_TOPIC_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (topic == nullptr) {std::cerr << "创建 Topic 失败!" << std::endl;return -1;}DDSSubscriber *subscriber = participant->create_subscriber(DDS_SUBSCRIBER_QOS_DEFAULT, nullptr, DDS_STATUS_MASK_NONE);if (subscriber == nullptr) {std::cerr << "创建 Subscriber 失败!" << std::endl;return -1;}DDSDataReader *reader = subscriber->create_datareader(topic, DDS_DATAREADER_QOS_DEFAULT, new MsgListener(), DDS_STATUS_MASK_ALL);if (reader == nullptr) {std::cerr << "创建 DataReader 失败!" << std::endl;return -1;}std::cout << "等待接收消息..." << std::endl;while (true) {DDS_Duration_t sleep_time = {1, 0};NDDSUtility::sleep(sleep_time);}participant->delete_contained_entities();DDSTheParticipantFactory->delete_participant(participant);return 0;
}

DDS和MQTT的区别

DDS(Data Distribution Service)MQTT(Message Queuing Telemetry Transport) 是两种用于分布式系统通信的协议或标准,它们在设计目标、特性和应用场景上有显著的区别。以下从多个方面对比它们:


1. 核心概念与设计目标

DDS

  • 是一种 实时分布式系统通信中间件标准,由 OMG(Object Management Group) 制定。
  • 提供 数据中心式的发布/订阅(Publish/Subscribe)模型,并且支持 点对点通信
  • 设计目标是为 低延迟、高吞吐量、实时性和可靠性 的系统提供支持,主要用于嵌入式系统、工业自动化、国防、航空航天、机器人、汽车等场景。
  • 强调 数据优先(Data-Centric),通过 Topic 直接共享数据,并支持复杂的 QoS(Quality of Service) 配置。

MQTT

  • 是一种轻量级的 消息队列协议,由 IBM 提出,后成为 OASIS 标准
  • 提供 发布/订阅(Publish/Subscribe)模型,但以 消息传输 为核心。
  • 设计目标是为 低带宽、高延迟、不可靠网络(如物联网设备)提供简单可靠的通信方式,广泛应用于物联网 (IoT)、智能家居、移动应用等场景。
  • 强调 消息优先(Message-Centric),通过 Broker 中转消息,设备之间并不直接通信。

2. 架构对比

DDS

  • 分布式架构:DDS 是去中心化的(Brokerless),无需中央服务器。所有参与者(如发布者、订阅者)可以直接通信。
  • 数据中心化:基于 Topic 共享数据,支持复杂的数据类型和强大的 QoS 配置。
  • 灵活性:支持动态发现(Dynamic Discovery),系统中的节点可以动态加入或离开。

MQTT

  • 中心化架构:MQTT 依赖于一个 Broker(中间服务器),所有消息都通过 Broker 转发。
  • 消息传输:基于 Topic 组织消息,消息是无状态的,数据类型简单(通常是字符串或 JSON)。
  • 设备简单性:客户端实现非常轻量,适用于低功耗设备。

3. QoS(服务质量)支持

DDS

  • DDS 提供了 22 种 QoS 策略,可以对实时性、可靠性、资源管理、延迟、传输顺序等进行精细控制。例如:
    • Deadline:设置数据传输的时间期限。
    • Latency Budget:定义通信的延迟预算。
    • Reliability:支持可靠传输(可靠模式)或最佳传输(不保证消息到达)。
    • Durability:支持数据的持久化,确保新订阅者可以获得历史数据。
  • DDS 的 QoS 机制非常灵活,适用于复杂场景。

MQTT

  • MQTT 提供了 3 种简单的 QoS 等级:
    1. QoS 0:消息最多发送一次,不保证到达。
    2. QoS 1:消息至少发送一次,可能重复。
    3. QoS 2:消息保证仅发送一次,且不会重复(最可靠)。
  • QoS 机制相对简单,主要适用于低复杂度的 IoT 场景。

4. 性能对比

DDS

  • 由于 DDS 是去中心化的,通信路径更短,通常具有 低延迟、高吞吐量 的特性。
  • 支持 实时性高可靠性,适合对延迟、抖动敏感的场景(如无人机控制、工业机器人)。
  • 性能依赖于网络和硬件环境,在高性能网络中表现极为出色。

MQTT

  • MQTT 由于采用中心化架构,通信性能受 Broker 服务器 的影响。
  • 适合 低带宽网络高延迟环境,但在实时性要求高的场景中表现不如 DDS。
  • 消耗资源少,非常适合低功耗设备。

5. 使用场景

DDS

  • 工业自动化:实时监控和控制(如工厂设备间通信)。
  • 航空航天:飞行器传感器数据共享、控制系统。
  • 机器人:机器人操作系统(如 ROS 2 的通信层基于 DDS)。
  • 汽车:自动驾驶汽车的传感器融合、车辆间通信。
  • 国防领域:分布式作战系统、军事仿真。

MQTT

  • 物联网 (IoT):智能家居、智能农业、环境监测。
  • 移动应用:即时消息、状态更新。
  • 远程监控:设备状态上传、警报通知。
  • 能源管理:智能电网监测。
  • 智能交通:传感器数据上传、车辆定位。

6. 易用性与学习曲线

DDS

  • DDS 功能强大,但 API 和配置相对复杂,学习曲线较陡。
  • 开发和调试 DDS 应用需要对 QoS 策略和分布式系统有较深入的了解。
  • 多数 DDS 实现(如 RTI Connext DDS)提供大量工具和文档,但仍需要较高的专业技能。

MQTT

  • MQTT 设计简单,易于上手,开发和部署快速。
  • 客户端实现轻量,适合嵌入式和低功耗设备。
  • 由于依赖 Broker,部署和管理集中化。

7. 开源实现

DDS

  • 常见的 DDS 实现包括:
    • RTI Connext DDS(商业版,功能强大)
    • OpenDDS(开源,Apache 2.0 许可证)
    • Eclipse Cyclone DDS(开源,Eclipse 基金会维护)
    • Fast-DDS(以前称为 FastRTPS,开源,eProsima 提供)
  • 这些实现通常支持多种编程语言(C++、Java、Python 等)。

MQTT

  • 常见的 MQTT Broker:
    • Mosquitto(开源,轻量级)
    • EMQX(开源,高性能,支持分布式部署)
    • HiveMQ(商业版,支持企业级功能)
    • VerneMQ(开源,支持大规模连接)
  • MQTT 客户端库也非常丰富,几乎支持所有主流语言。

8. 总结对比

特性DDSMQTT
架构去中心化(Brokerless)中心化(需要 Broker)
实时性高实时性,低延迟针对低带宽、高延迟环境
QoS复杂且灵活(22 种 QoS 策略)简单(3 种 QoS 等级)
适用场景工业自动化、航空航天、机器人等IoT、智能家居、移动应用等
学习曲线较陡,需要较高专业技能简单,快速上手
性能高吞吐量、低延迟适合低功耗和高延迟网络

总结

  • 如果你的项目需要 实时性、高可靠性、复杂 QoS,并且运行在高性能网络中,例如工业自动化或机器人控制,DDS 是更好的选择
  • 如果你的项目需要 轻量、低功耗、简单消息传递,运行在低带宽或不可靠的网络中,例如物联网设备或智能家居,MQTT 是更适合的选择

相关文章:

C++中间件DDS介绍

C DDS 库简介 DDS&#xff08;Data Distribution Service&#xff09; 是一种用于实时分布式系统通信的中间件标准&#xff0c;由 OMG&#xff08;Object Management Group&#xff09; 提出。它是一种发布/订阅&#xff08;Publish/Subscribe&#xff09;模式的数据通信框架&…...

自动化之ansible(二)

一、ansible中playbook&#xff08;剧本&#xff09; 官方文档&#xff1a; Ansible playbooks — Ansible Community Documentation 1、playbook的基本结构 一个基本的playbook由以下几个主要部分组成 hosts: 定义要执行任务的主机组或主机。 become: 是否需要使用超级用户…...

QSNCTF-WEB做题记录

第一题&#xff0c;文章管理系统 来自 <天狩CTF竞赛平台> 描述&#xff1a;这是我们的文章管理系统&#xff0c;快来看看有什么漏洞可以拿到FLAG吧&#xff1f;注意&#xff1a;可能有个假FLAG哦 1&#xff0c;首先观察题目网站的结构和特征 这个一个文件管理系统&#x…...

Ruoyi-Vue 3.8.7集成积木报表JmReport和积木大屏JimuBI

Ruoyi-Vue 3.8.7集成积木报表JmReport和积木大屏JimuBI 一、版本 RuoYi-Vue版本&#xff1a;v3.8.7 JMreport报表版本&#xff1a; v1.9.4 JimuBI大屏版本&#xff1a;V1.9.4 二、数据库 积木数据库sql 下载后&#xff0c;使用数据库管理工具执行sql脚本&#xff0c;将需…...

OSPF(开放路径最短优先)

ospf优先级&#xff1a;内部优先级默认为10&#xff0c;外部优先级默认为150 1.ospf的三张表 &#xff08;1&#xff09;邻居表 <记录邻居状态和关系> &#xff08;2&#xff09;拓扑表 <链路状态数据库> &#xff08;3&#xff09;路由表 <对链路状态数据库进…...

请谈谈 Vue 中的响应式原理,如何实现?

一、Vue2响应式原理&#xff1a;Object.defineProperty的利与弊 实现原理&#xff1a; // 数据劫持核心实现 function defineReactive(obj, key, val) {const dep new Dep(); // 依赖收集容器Object.defineProperty(obj, key, {get() {if (Dep.target) { // 当前Watcher实例…...

亲测可用,IDEA中使用满血版DeepSeek R1!支持深度思考!免费!免配置!

作者&#xff1a;程序员 Hollis 之前介绍过在IDEA中使用DeepSeek的方案&#xff0c;但是很多人表示还是用的不够爽&#xff0c;比如用CodeChat的方案&#xff0c;只支持V3版本&#xff0c;不支持带推理的R1。想要配置R1的话有特别的麻烦。 那么&#xff0c;今天&#xff0c;给…...

jvm中各个参数的理解

MEMORY - MANAGERS 定义 MEMORY - MANAGERS即内存管理器&#xff0c;它是操作系统或软件系统中负责管理计算机内存资源的组件。从本质上来说&#xff0c;它是一种软件机制&#xff0c;旨在协调计算机系统中内存的分配、使用和回收等操作&#xff0c;确保系统能够高效、稳定地…...

【队列】循环队列(Circular Queue)详解

文章目录 一、循环队列简介二、循环队列的判空和判满三、循环队列的实现leetcode 622. 设计循环队列 一、循环队列简介 在实际开发中&#xff0c;队列是一种常用的数据结构&#xff0c;而循环队列&#xff08;Circular Queue&#xff09;则一般是一种基于数组实现的队列&#x…...

Deepseek快速做PPT

背景: DeepSeek大纲生成 → Kimi结构化排版 → 数据审查,细节调整 DeepSeek 拥有深度思考能力,擅长逻辑构建与内容生成,它会根据我们的问题进行思考,其深度思考能力当前测试下来,不愧为国内No.1,而且还会把中间的思考过程展示出来,大多时候会给出很多我们意想不到的思…...

DeepSeek掀起推理服务器新风暴,AI应用迎来变革转折点?

AI 浪潮下&#xff0c;推理服务器崭露头角 在科技飞速发展的当下&#xff0c;AI 是耀眼明星&#xff0c;席卷各行业&#xff0c;深刻改变生活与工作模式&#xff0c;从语音助手到医疗诊断、金融风险预测&#xff0c;AI 无处不在。其发展分数据收集整理、模型训练、推理应用三个…...

离线部署大模型:ollama+deepseek+open-webui

ollama 是一个开源的本地大语言模型运行框架&#xff0c;它提供了非常简单便捷的使用形式&#xff0c;让用户可以十分方便的在本地机器上部署和运行大型语言模型&#xff0c;从而实现免费离线的方式使用 LLM 能力&#xff0c;并确保私有数据的隐私和安全性。 1 ollama 安装 o…...

深入解析浏览器渲染全流程:从URL输入到页面渲染的底层原理与性能优化(附实战代码)

本文以https://example.com为例&#xff0c;逐层剖析浏览器从输入URL到页面渲染的完整链路&#xff0c;涵盖DNS解析、TCP/TLS握手、HTTP请求、DOM/CSSOM构建等核心阶段&#xff0c;结合代码示例与性能调优技巧&#xff0c;助你掌握浏览器底层运行机制。 一、导航阶段&#xff1…...

现代游戏UI架构深度解析——以UIController为核心的模块化界面管理系统

一、架构全景与设计哲学 本文将以重构后的UIController为核心&#xff0c;深入探讨Unity引擎下的高效UI管理方案。该体系采用"分层-分治"设计理念&#xff0c;通过界面生命周期管理、动态适配策略、资源优化机制三个维度的协同工作&#xff0c;构建了适应复杂交互需…...

Vue 项目中逐步引入 TypeScript 的类型检查

在现有的 Vue 项目中逐步引入 TypeScript 的类型检查 本文源于一道面试题&#xff1a;注&#xff1a;两种问法一个意思哈&#xff01;&#xff01; 问题一&#xff1a;“ 老项目Js写的&#xff0c;如何轻量方式享受 ts 类型&#xff1f;” 问题二&#xff1a;“如何 在现有的 …...

Git企业开发

Git&#xff08;版本控制器&#xff09; 在我们对于文档进行操作的时候&#xff0c;很多时候可能会出现多个文档&#xff0c;对这些文档进行多个版本的保存和记录就变成必要的。通俗的讲&#xff0c;就是记录每次的修改和记录版本迭代的管理系统。目前最主流的版本控制器就是G…...

DeepSeek预测25考研分数线

25考研分数马上要出了。 目前&#xff0c;多所大学已经陆续给出了分数查分时间&#xff0c;综合往年情况来看&#xff0c;每年的查分时间一般集中在2月底。 等待出成绩的日子&#xff0c;学子们的心情是万分焦急&#xff0c;小编用最近爆火的“活人感”十足的DeepSeek帮大家预…...

备战蓝桥杯 -牛客

习题-[NOIP2006]明明的随机数 1046-习题-[NOIP2006]明明的随机数_2021秋季算法入门班第一章习题&#xff1a;模拟、枚举、贪心 思路&#xff1a;这道题用stl的set&#xff0c;今天写这道题复习了一下set的用法&#xff1a; s.find(a) s.end()的意思是判断元素a是否存在于集…...

基于springboot校园健康系统的设计与实现(源码+文档)

大家好我是风歌&#xff0c;今天要和大家聊的是一款基于springboot的园健康系统的设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 基于springboot校园健康系统的设计与实现的主要使用者管理员具有最高的权限&#xff0c;通…...

出现 [ app.json 文件内容错误] app.json: 在项目根目录未找到 app.json (env: Windows,mp 解决方法

目录 1. 问题所示2. 原理分析3. 解决方法1. 问题所示 hbuilder X 执行代码的时候出现如下所示 [ app.json 文件内容错误] app.json: 在项目根目录未找到 app.json (env: Windows,mp,1.06.2412050; lib:...

设计模式教程:责任链模式(Chain of Responsibility Pattern)

责任链模式&#xff08;Chain of Responsibility Pattern&#xff09;是一种常用的设计模式&#xff0c;它属于行为型模式&#xff0c;主要解决的是多个对象处理一个请求时&#xff0c;如何解耦请求的发送者和接收者&#xff0c;以及如何将请求的处理职责分配给不同的对象。 1…...

【YOLOv8】损失函数

学习视频&#xff1a; yolov8 | 损失函数 之 5、类别损失_哔哩哔哩_bilibili yolov8 | 损失函数 之 6、定位损失 CIoU DFL_哔哩哔哩_bilibili 2.13、yolov8损失函数_哔哩哔哩_bilibili YOLOv8 的损失函数由类别损失和定位损失构成 类别损失&#xff1a;BCE Loss 定位损失…...

ollama修改监听ip: 0.0.0.0

确认Ollama绑定IP地址 默认情况下&#xff0c;Ollama可能仅监听本地回环地址&#xff08;127.0.0.1&#xff09;。要允许外部访问&#xff0c;需将其配置为监听所有IP&#xff08;0.0.0.0&#xff09;或指定IP&#xff08;如10…19&#xff09;。 修改启动命令&#xff08;推荐…...

【Linux】【网络】Libevent 内部实现简略版

【Linux】【网络】Libevent 内部实现简略版 1 event_base结构–>相当于Reactor 在使用libevent之前&#xff0c;就必须先创建这个结构。 以epoll为例&#xff1a; 1.1evbase void* evbase-->epollop结构体&#xff08;以epoll为例&#xff09; libevent通过一个void…...

计算机网络抄手 运输层

一、运输层协议概述 1. 进程之间的通信 从通信和信息处理的角度看&#xff0c;运输层向它上面的应用层提供通信服务&#xff0c;它属于面向通信部分的最高层&#xff0c;同时也是用户功能中的最低层。当网络边缘部分的两台主机使用网络核心部分的功能进行端到端的通信时&…...

MATLAB图像处理:图像分割方法

图像分割将图像划分为具有特定意义的子区域&#xff0c;是目标检测、医学影像分析、自动驾驶等领域的核心预处理步骤。本文讲解阈值分割、边缘检测、区域生长、聚类分割、基于图的方法等经典与前沿技术&#xff0c;提供MATLAB代码实现。 目录 1. 图像分割基础 2. 经典分割方…...

【机器学习】线性回归 多元线性回归

多元线性回归 V1.1多元线性回归一元线性回归与多元线性回归多元线性回归模型的误差衡量多元线性回归的最优解多元线性回归的解析解&#xff08;标准数学解法&#xff09;多元线性回归的解析解公式分析 多元线性回归的搜索解法 V1.1 加入链接会影响文章推荐权重&#xff0c;阅读…...

【VSCode】MicroPython环境配置

【VSCode】MicroPython环境配置 RT-Thread MicroPython 插件安装MicroPython 库文件配置结束语 RT-Thread MicroPython 插件安装 在 VSCode 拓展中搜索 “RT-Thread MicroPython” 并安装&#xff0c;详细配置步骤&#xff08;修改 VSCode 默认终端、MicroPython 代码补全&…...

【python】网页批量转PDF

安装wkhtmltopdf 网站&#xff1a;wkhtmltopdf wkhtmltopdf http://www.baidu.com/ D:website1.pdf 安装pdfkit库 pip install pdfkit 批量转换代码 import os import pdfkit path_wkthmltopdf rE:\Program Files\wkhtmltopdf\bin\wkhtmltopdf.exe config pdfkit.configu…...

基于Flask的租房信息可视化系统的设计与实现

【Flask】基于Flask的租房信息可视化系统的设计与实现&#xff08;完整系统源码开发笔记详细部署教程&#xff09;✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 随着互联网的快速发展&#xff0c;租房市场日益繁荣&#xff0c;信息量急剧增加&#xff…...