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. 创建 DomainParticipant
    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;
    }

    // 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. 创建 Topic
    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;
    }

    // 4. 创建 Publisher 和 DataWriter
    DDSPublisher *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 是更适合的选择

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/973160.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大白话实战Gateway

网关功能 网关在分布式系统中起了什么作用?参考下图: 前端想要访问业务访问,就需要知道各个访问的地址,而业务集群服务有很多,前端需要记录非常多的服务器地址,这种情况下,我们需要对整个业务集群做一个整体屏蔽,这个时候就引入Gateway网关,它就是所有服务的请求入…

用大内存主机下载Visual Studio

用一台内存达到128G的主机下载Visual Studio 2022&#xff0c;用的是公司网络。下载速度让我吃了一惊&#xff0c;没人用网络了&#xff1f;还是网站提速了&#xff1f;以前最大只能达到5MB/秒。记录这段经历&#xff0c;是用来分析公司网络用的......

【C++语言】string 类

一、为什么要学习 string 类 C语言中&#xff0c;字符串是以 “\0” 结尾的一些字符的集合&#xff0c;为了操作方便&#xff0c;C标准库中提供了一些 str 系列的库函数&#xff0c;但是这些库函数与字符串是分离开的&#xff0c;不太符合 OOP 的思想&#xff0c;而且底层空间需…

深度学习-123-综述之AI人工智能与DL深度学习简史1956到2024

文章目录 1 AI与深度学习的简史1.1 人工智能的诞生(1956)1.2 早期人工神经网络(1940-1960年代)1.3 多层感知器MLP(1960年代)1.4 反向传播(1970-1980年代)1.5 第二次黑暗时代(1990-2000年代)1.6 深度学习的复兴(21世纪末至今)1.6.1 CNN卷积神经网络(1980-2010)1.6.2 RNN递归神经…

解决本地模拟IP的DHCP冲突问题

解决 DHCP 冲突导致的多 IP 绑定失效问题 前言 续接上一篇在本机上模拟IP地址。 在实际操作中&#xff0c;如果本机原有 IP&#xff08;如 192.168.2.7&#xff09;是通过 DHCP 自动获取的&#xff0c;直接添加新 IP&#xff08;如 10.0.11.11&#xff09;可能会导致 DHCP 服…

基于Llama 3.2-Vision的医学报告生成

记录运用大模型解决医学报告实例&#xff0c;仅介绍本地调用的情况。 前情提要 已安装 Python 显存不少于8G&#xff08;8G设备上测试成功&#xff0c;其他环境可以自行测试&#xff09;。 需要安装Ollama (Ollama 是一个允许在本地运行多模态模型的平台)。 方式1&#xff1…

DeepSeek预测25考研分数线

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

DeepSeek赋能智慧文旅:新一代解决方案,重构文旅发展的底层逻辑

DeepSeek作为一款前沿的人工智能大模型&#xff0c;凭借其强大的多模态理解、知识推理和内容生成能力&#xff0c;正在重构文旅产业的发展逻辑&#xff0c;推动行业从传统的经验驱动向数据驱动、从人力密集型向智能协同型转变。 一、智能服务重构&#xff1a;打造全域感知的智…

【Python爬虫(26)】Python爬虫进阶:数据清洗与预处理的魔法秘籍

【Python爬虫】专栏简介&#xff1a;本专栏是 Python 爬虫领域的集大成之作&#xff0c;共 100 章节。从 Python 基础语法、爬虫入门知识讲起&#xff0c;深入探讨反爬虫、多线程、分布式等进阶技术。以大量实例为支撑&#xff0c;覆盖网页、图片、音频等各类数据爬取&#xff…

支持批量导出的软件,效率拉满!

今天给大家分享一款超实用的软件&#xff0c;它能帮你批量导出PPT里的图片&#xff0c;简直是提升工作效率的神器&#xff01; PPT转jpg PPT逐页导出为图片 这款软件超级简单易用&#xff0c;打开就能直接上手&#xff0c;不需要复杂的设置。 这个软件有三种功能&#xff0c; …

论文笔记(七十二)Reward Centering(二)

Reward Centering&#xff08;二&#xff09; 文章概括摘要2 简单的奖励中心 文章概括 引用&#xff1a; article{naik2024reward,title{Reward Centering},author{Naik, Abhishek and Wan, Yi and Tomar, Manan and Sutton, Richard S},journal{arXiv preprint arXiv:2405.0…

Jmeter连接数据库、逻辑控制器、定时器

Jmeter直连数据库 直接数据库的使用场景 直连数据库的关键配置 添加MYSQL驱动Jar包 方式一&#xff1a;在测试计划面板点击“浏览”按钮&#xff0c;将你的JDBC驱动添加进来 方式二&#xff1a;将MySQL驱动jar包放入到lib/ext目录下&#xff0c;重启JMeter 配置数据库连接信…

ORM框架详解:为什么不直接写SQL?

想象一下&#xff0c;你正在开发一个小型的在线书店应用。你需要存储书籍信息、用户数据和订单记录。作为一个初学者&#xff0c;你可能会想&#xff1a;“我已经学会了SQL&#xff0c;为什么还要使用ORM框架呢&#xff1f;直接写SQL语句不是更简单、更直接吗&#xff1f;” 如…

RT-Thread+STM32L475VET6实现红外遥控实验

文章目录 前言一、板载资源介绍二、具体步骤1. 确定红外接收头引脚编号2. 下载infrared软件包3. 配置infrared软件包4. 打开STM32CubeMX进行相关配置4.1 使用外部高速时钟&#xff0c;并修改时钟树4.2 打开定时器16(定时器根据自己需求调整)4.3 打开串口4.4 生成工程 5. 打开HW…

推荐一个github star45k+进阶的java项目及知识的网站

mall是github上star 45k的一个java项目 mall项目是一套电商系统&#xff0c;包括前台商城系统及后台管理系统&#xff0c;基于SpringBootMyBatis实现&#xff0c;采用Docker容器化部署。 前台商城系统包含首页门户、商品推荐、商品搜索、商品展示、购物车、订单流程、会员中心…

pyside6学习专栏(二):程序图像资源的加载方式

pyside6中的QLabel控件可以加载图像和gif动画&#xff0c;可以直接从外部文件加载&#xff0c;也可以从QRC类型的文件(实际是一脚本文件)经编绎生成对应的资源.PY模块文件(就是将qrc文本中指定的资源文件的16制内容写入.py文件)来使用&#xff0c;本文对两种方式作了一简单的示…

项目管理的核心是什么?

项目管理不仅仅是按照一定的计划进行任务的执行&#xff0c;更重要的是如何在面对复杂和动态的环境下&#xff0c;保证项目顺利进行并达到预期的结果。它的核心在于高效的资源配置、团队的合作与协调、风险管理及变更管理。在这些关键因素的支持下&#xff0c;项目能够高效地从…

10分钟上手DeepSeek开发:SpringBoot + Vue2快速构建AI对话系统

作者&#xff1a;后端小肥肠 目录 1. 前言 为什么选择DeepSeek&#xff1f; 本文技术栈 2. 环境准备 2.1. 后端项目初始化 2.2. 前端项目初始化 3. 后端服务开发 3.1. 配置文件 3.2. 核心服务实现 4. 前端服务开发 4.1. 聊天组件ChatWindow.vue开发 5. 效果展示及源…

【C++第二十章】红黑树

【C第二十章】红黑树 红黑树介绍&#x1f9d0; 红黑树是一种自平衡的二叉搜索树&#xff0c;通过颜色标记和特定规则保持树的平衡性&#xff0c;从而在动态插入、删除等操作中维持较高的效率。它的最长路径不会超过最短路径的两倍&#xff0c;它的查找效率比AVL树更慢(对于CPU…

Docker+Dify部署DeepSeek-r1本地知识库

安装配置Docker Desktop 软件下载 Docker Desktop版本:4.38.0.181591 Docker Desktop下载地址:Docker: Accelerated Container Application Development 或者从这里下载:DockerDesktop-4.38.0.181591资源-CSDN文库 点击图下所示位置,下载windows-AMD64版本软件 启用Hy…