基于开源库编写MQTT通讯

目录

  • 1. MQTT是什么?
  • 2. 开发交互UI
  • 3. 服务器核心代码
  • 4. 客户端核心代码
  • 5. 消息订阅与发布
  • 6. 通讯测试
  • 7. MQTT与PLC通讯
  • 最后. 核心总结


1. MQTT是什么?

MQTT(Message Queuing Terlemetry Transport)消息队列遥测协议;是一种轻量级的发布/订阅消息传输协议,专为IOT、低带宽、高延迟的网络环境设计,具有高效、低耗、海量设备连接特性。

1.通讯原理

  • 发布(Publisher):发布消息到特定主题(Topic)
  • 订阅(Subscriber):订阅主题接收消息
  • 代理(Broker):消息路由管理,接收发布的消息,颁发给所有订阅者

示例:传感器(Publisher)发送消息到主题(Topic),代理(Broker)接收消息并检查该主题的所有客户端,订阅者(Subscriber)实时接收消息

2.消息结构

  • 主题(topic)+负载(payload)

3.必要条件

  • MQTT Broker: 消息代理服务器
  • 客户端库: 设备或应用需集成MQTT客户端
  • 端口: 默认非加密端口 1883,加密端口 8883(TLS/SSL)
  • 连接认证: 支持用户名/密码、客户端证书等认证方式
  • QoS服务质量/策略
QoS等级描述适用场景
0最多一次(无确认)实时性高,允许丢数据
1至少一次(需确认)数据需可靠但不重复
2精确一次(握手确认)关键数据,严格不重复

4. 关键特性
遗嘱消息(Last Will)

  • 设备异常断开时,Broker 自动发布预设消息(如“设备离线”),通知其他客户端。

保留消息(Retained)

  • Broker 保存主题的最新消息,新订阅者首次连接时立即获取。

主题层级(Topic Hierarchy)

  • 支持多级通配符(+ 单层,# 多层)
  • 例如:home/floor1/temperature; home/+/status(匹配所有楼层状态)

5.应用场景

  • 移动设备远程监控(AGV状态上报)
  • 跨厂区数据汇聚(通过云平台中转)

2. 开发交互UI

创建WindowsFormAPP项目,NuGet安装MQTTnet开源库(项目-属性-框架;需与程序包的依赖项一致<否则安装错误>:PM> NuGet\Install-Package MQTTnet -Version 2.8.2

控件

  • label、TextBox、ComboBox、Button
  • ListView(Dock=停靠<视图=Details;小Imagelist=Imagelist1;编辑列=Infoname、Content>)

容器

  • SplitContainer(Orienting=垂直;SplitterWith=1;BorderStyle=边框)

组件

  • Imagelist(添加图像)
  • timer(Enabled=true、Interval=1000)

状态

  • StatusStrip(系统时间、连接数量、版本说明)

Server端

  • 窗体设置(Text=标题、Font=字体、StartPosition=位置、FormBoardStyle=边框)
  • 给定服务IP,固定端口号,设置开始、停止服务、快捷打开客户端按钮
  • 设置日志消息显示窗口;设置状态栏

在这里插入图片描述

Client端

  • 技巧:复制FrmServer(修改窗口、Designer代码)
  • 容器:在SP1的Panel2添加SplitContainer2(Listview放在SP2的Panel1中);
  • 设置SP1的FixedPanel的Panel1不动;设置SP2的FixedPanel的Panel2不动
  • 客户端可发布主题消息
  • 设置连接、订阅、取消订阅、发布主题按钮;
  • 主题信息、给定QoS策略

在这里插入图片描述在这里插入图片描述


3. 服务器核心代码

初始化-public FrmServer(){…}

  • 获取IP集合(Dns.GetHostAddresses)
  • 绑定控件(cmb_iplist.DataSource、.SelectedIndex)

创建服务器对象(IMqttServer)(->服务启动按钮点击事件)

  • 创建服务器配置 _ var optionsBuilder = new MqttServerOptionsBuilder()
    – 验证用户密码_ .WithConnectionValidator(c =>...
  • 实例化服务对象_mqttServer = new MqttFactory().CreateMqttServer();
  • 创建MQTT事件方法_mqttServer.ClientConnected += MqttServer_ClientConnected;
    – 方法日志显示_this.AddLog(0, "MQTT客户端已连接" + "ClietID:" + e.ClientId.Length);
  • 启动服务_mqttServer.StartAsync(optionsBuilder.Build());

日志对象

  • 创建委托_public delegate void AddlogDelegate(int index, string info);
  • 委托方法_private void AddlogMothod(int index, string info){...}
  • 委托对象_private AddlogDelegate AddLog;
  • 对象绑定方法 _this.AddLog = this.AddlogMothod;
  • 对象应用eg:_ this.AddLog(0, "MQTT服务端已停止");

状态栏

  • 系统时间_this.tss_time.Text = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
  • 连接个数(连接/断开事件更新)_this.tss_connnum.Text = mqttServer.GetClientSessionsStatusAsync().Result.Count.ToString();

其他

  • 停止服务_mqttServer.StopAsync();
  • 清空日志_this.lst_info.Items.Clear();
  • 打开客户端(注意“重新生成解决方案”)_new FrmClient().Show();
  • 当前时间_DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
  • 关闭窗体停止服务

4. 客户端核心代码

初始化

  • 获取IP集合(Dns.GetHostAddresses)
  • 绑定控件(cmb_iplist.DataSource、.SelectedIndex)

快捷打开客户端,ServerIP、Port自动填写

  • 创建带参数的初始化构造方法_ public FrmClient(string ip, string port)

创建客户端对象(IMqttClient)(->客户端连接按钮点击事件)

  • 创建客户端配置_ var option = new MqttClientOptions() { ClientId = Guid.NewGuid().ToString("D") };
  • 创建通道配置_ option.ChannelOptions = new MqttClientTcpOptions()
  • 是否启用账户_ if (this.chk_isuserpwd.Checked)
  • 创建会话配置_option.CleanSession = true;
  • 创建客户端对象_mqttClient = new MqttFactory().CreateMqttClient();
  • 连接服务器_ mqttClient.ConnectAsync(option);
  • 绑定事件方法(添加日志)_ mqttClient.Connected += mqttClient_Connected;

日志显示

  • 创建委托方法_与服务端一致
  • 创建委托对象_与服务端一致
  • 初始化对象绑定方法_与服务端一致
  • 对象引用eg_与服务端一致

其他

  • 断开连接_mqttClient.DisconnectAsync();

5. 消息订阅与发布

客户端可以正确连接后,只有订阅与发布消息,才算真正进行数据通讯;消息的订阅和发布均在客户端进行,服务端只需提供一个服务供客户连接(桥梁的作用);小节解释消息订阅发布的核心代码。
QoS服务策略(下拉框获取)

  • QoS服务策略(枚举类型)_this.cmb_qos_pub.DataSource = Enum.GetNames(typeof(MqttQualityOfServiceLevel));

消息订阅

  • 订阅主题_mqttClient.SubscribeAsync(new List<TopicFilter>(){...});1

消息取消订阅

  • 取消订阅_mqttClient.UnsubscribeAsync(this.txt_topic_sub.Text);

消息发布

  • 创建消息对象_ var msg = new MqttApplicationMessage(){...} 2
  • 发布消息_mqttClient.PublishAsync(msg);

6. 通讯测试

在这里插入图片描述


7. MQTT与PLC通讯

将msg对象中的Payload更改为PLC的寄存器即可
自动发布

  • 添加定时器timer1_设置频率,事件(消息内容,自动发布)
  • 连接成功时启动定时器_ this.timer1.Enabled = true;

消息内容

  • 添加引用(西门子通讯库)
  • 创建PLC对象
  • 读取PLC变量
  • 添加到msg中的Payload中_Payload = Encoding.UTF8.GetBytes(plcmsg),

在这里插入图片描述


最后. 核心总结

1. 开发实现
服务端:

  • 功能:IP/端口配置、启动/停止服务、连接监控
  • 初始化 MqttServer 对象
  • 处理连接/断开事件(日志记录、状态更新)

客户端:

  • 功能:连接/断开、订阅/取消主题、消息发布
  • 配置 MqttClient(IP、端口、认证)
  • 实现订阅 (SubscribeAsync) /发布 (PublishAsync)

2. 进阶应用

PLC 集成:

  • 自动发布:定时器读取 PLC 数据并推送
  • 数据格式:Payload 封装寄存器值(如 Siemens PLC 数据)

异常测试:

  • 基础测试:服务端启停、客户端连接/断开
  • 消息流验证:订阅发布功能、QoS 策略生效
  • 异常测试:网络断开重连、遗嘱消息触发

附:关键代码片段
服务端启动:

var options = new MqttServerOptionsBuilder().WithDefaultPort(1883).Build();
mqttServer = new MqttFactory().CreateMqttServer();
mqttServer.StartAsync(options);

客户端发布消息:

var msg = new MqttApplicationMessage {Topic = "sensor/temp",Payload = Encoding.UTF8.GetBytes("25℃"),QoS = MqttQualityOfServiceLevel.AtLeastOnce
};
client.PublishAsync(msg);

PLC 数据读取:

var plcValue = SiemensPLC.Read("DB1.DBD0"); // 读取浮点数
var payload = $"{{\"temperature\": {plcValue}}}";

源码链接


  1. 创建TopicFilter对象_new TopicFilter(this.txt_topic_sub.Text, (MqttQualityOfServiceLevel)Enum.Parse(typeof(MqttQualityOfServiceLevel),this.cmb_qos_sub.Text)) ↩︎

  2. 消息对象赋值_ Topic = this.txt_topic_pub.Text, QualityOfServiceLevel =(MqttQualityOfServiceLevel)Enum.Parse(typeof(MqttQualityOfServiceLevel), this.cmb_qos_pub.Text), Payload = Encoding.UTF8.GetBytes(this.txt_msg_pub.Text), Retain = false, ↩︎

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vector习题

完数和盈数 题目 完数VS盈数_牛客题霸_牛客网 一个数如果恰好等于它的各因子(该数本身除外)之和&#xff0c;如&#xff1a;6321。则称其为“完数”&#xff1b;若因子之和大于该数&#xff0c;则称其为“盈数”。 求出2到60之间所有“完数”和“盈数”。 输入描述&#xff…

vscode通过ssh远程连接(linux系统)不能跳转问题

1.问题描述 unbantu中的vscode能够通过函数跳转到函数定义&#xff0c;而windows通过ssh连接unbantu的vscode却无法跳转 2.原因&#xff1a; 主要原因是这里缺少插件&#xff0c;这里是unbantu给主机的服务器&#xff0c;与ubantu本地vscode插件相互独立&#xff0c;能否跳转…

神经网络 - 激活函数(Swish函数、GELU函数)

一、Swish 函数 Swish 函数是一种较新的激活函数&#xff0c;由 Ramachandran 等人在 2017 年提出&#xff0c;其数学表达式通常为 其中 σ(x) 是 Sigmoid 函数&#xff08;Logistic 函数&#xff09;。 如何理解 Swish 函数 自门控特性 Swish 函数可以看作是对输入 x 进行“…

安全运营的“黄金4小时“:如何突破告警疲劳困局

在当今复杂多变的网络安全环境中&#xff0c;安全团队面临着前所未有的挑战。尤其是面对高级持续性威胁&#xff08;APT&#xff09;时&#xff0c;最初的“黄金4小时”成为决定成败的关键窗口。在这段时间内&#xff0c;快速而准确地响应可以极大地降低损失&#xff0c;然而&a…

【Pytest】setup和teardown的四个级别

文章目录 1.setup和teardown简介2.模块级别的 setup 和 teardown3.函数级别的 setup 和 teardown4.方法级别的 setup 和 teardown5.类级别的 setup 和 teardown 1.setup和teardown简介 在 pytest 中&#xff0c;setup 和 teardown 用于在测试用例执行前后执行一些准备和清理操…

傅里叶分析

傅里叶分析之掐死教程&#xff08;完整版&#xff09;更新于2014.06.06 要让读者在不看任何数学公式的情况下理解傅里叶分析。 傅里叶分析不仅仅是一个数学工具&#xff0c;更是一种可以彻底颠覆一个人以前世界观的思维模式。但不幸的是&#xff0c;傅里叶分析的公式看起来太复…

matlab 四维数据可视化(已解决)

虽然这不是传统意义上的“4维可视化”&#xff0c;但你可以通过在三维空间中表示两个维度来间接展示4维数据。例如&#xff0c;你可以使用颜色来表示第四个维度。 clc clear close all% 假设X, Y, Z为你的三维数据&#xff0c;C为第四维数据 X rand(100, 1); Y rand(100, 1);…

MAC 本地搭建部署 dify(含 github访问超时+Docker镜像源拉取超时解决方案)

目录 一、什么是 dify&#xff1f; 二、安装 docker 1. 什么是 docker&#xff1f; 2. docker下载地址 三、安装 dify 1. dify下载地址 2.可能遇到问题一&#xff1a; github访问超时 3.下载后完成解压 4.进入到 cmd 终端环境&#xff0c;执行下面三个命令 5.可能遇到…

rustup-init.exe 安装缓慢的解决办法

首先在rust官网下载安装程序&#xff0c;官网下载的 rustup-init.exe 下载慢&#xff0c;安装慢&#xff0c;或者直接卡死。 下载安装程序在本地&#xff0c;使用国内镜像加速 Rust 更新与下载。 使用国内镜像源&#xff1a;在 rustup-init.exe 程序文件夹下使用 PowerShell 中…

OpenAI发布GPT-4.5:功能非常特殊,推理很贵

今天凌晨4点&#xff0c;OpenAI进行了在线技术直播&#xff0c;发布了最新模型GPT-4.5。 GPT-4.5与之前的模型相比&#xff0c;本次最大的亮点是加上了“情商”&#xff0c;这也是目前所有大模型最缺、最难的功能。 此外&#xff0c;GPT-4.5 在SimpleQA上的测试数据显示&…

unity pico开发二:连接头盔,配置手柄按键事件

文章目录 导入UnityXR Interaction ToolKit构建基础内容 导入UnityXR Interaction ToolKit 检查一下packagemanager&#xff0c;unityxr interactionToolkit是否自动导入 我们需要升级到一个不超过3.x的版本&#xff0c;因为pico还不支持3.x的内容 然后右侧samples里导入初始…

k8s架构及服务详解

目录 1.1.容器是什么1.2.Namespace1.3.rootfs5.1.Service介绍5.1.1.Serice简介 5.1.1.1什么是Service5.1.1.2.Service的创建5.1.1.3.检测服务5.1.1.4.在运行的容器中远程执行命令 5.2.连接集群外部的服务 5.2.1.介绍服务endpoint5.2.2.手动配置服务的endpoint5.2.3.为外部服务…

【含文档+PPT+源码】基于SpringBoot和Vue的编程学习系统

项目介绍 本课程演示的是一款 基于SpringBoot和Vue的编程学习系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的 Java 学习者。 1.包含&#xff1a;项目源码、项目文档、数据库脚本、软件工具等所有资料 2.带你从零开始部署运行本套系统 3.该…

【FL0090】基于SSM和微信小程序的球馆预约系统

&#x1f9d1;‍&#x1f4bb;博主介绍&#x1f9d1;‍&#x1f4bb; 全网粉丝10W,CSDN全栈领域优质创作者&#xff0c;博客之星、掘金/知乎/b站/华为云/阿里云等平台优质作者、专注于Java、小程序/APP、python、大数据等技术领域和毕业项目实战&#xff0c;以及程序定制化开发…

macos下myslq图形化工具之Sequel Ace

什么是Sequel Ace 官方github&#xff1a;https://github.com/Sequel-Ace/Sequel-Ace Sequel Ace 是一款快速、易于使用的 Mac 数据库管理应用程序&#xff0c;用于处理 MySQL 和 MariaDB 数据库。 Sequel Ace 是一款开源项目&#xff0c;采用 MIT 许可证。用户可以通过 Ope…

1.2.2 使用Maven方式构建Spring Boot项目

本次实战通过Maven方式构建了一个Spring Boot项目&#xff0c;实现了简单的Web应用。首先&#xff0c;创建了Maven项目并设置好项目名称、位置、构建系统和JDK等。接着&#xff0c;添加了Spring Boot的父项目依赖和web、thymeleaf起步依赖。然后&#xff0c;创建了项目启动类He…

Java【多线程】(2)线程属性与线程安全

目录 1.前言 2.正文 2.1线程的进阶实现 2.2线程的核心属性 2.3线程安全 2.3.1线程安全问题的原因 2.3.2加锁和互斥 2.3.3可重入&#xff08;如何自己实现可重入锁&#xff09; 2.4.4死锁&#xff08;三种情况&#xff09; 2.4.4.1第一种情况 2.4.4.2第二种情况 2.4…

[IP] DDR_FIFO(DDR3 用户FIFO接口)

IP(DDR_FIFO)将DDR3 IP的用户侧复杂接口修改为简易的FIFO接口&#xff0c;用户侧更加简易例化使用MIG 核 IP介绍 c0_xx (连接DDR app接口) 此IP 仅需根据MIG配置进行有限修改&#xff0c;即可使用&#xff01; 关于IP详细使用说明&#xff0c;参考IP datasheet&#xff01; 示…

USRP6330-通用软件无线电平台

1、产品描述 USRP6330平台以XILINX XCZU15EG SOC处理器为核心&#xff0c;搭配两片ADI ADRV9026射频集成芯片&#xff0c;提供了瞬时带宽高达200MHz的8收8发射频通道。通过驯服的高精度GPSDO时钟参考方案&#xff0c;USRP可以支持高性能的MIMO通信系统&#xff0c;提供了部署大…

P8615 [蓝桥杯 2014 国 C] 拼接平方数--substr、to_string、stoi

P8615 [蓝桥杯 2014 国 C] 拼接平方数--substr、to_string、stoi 题目 解析介绍一下这三个函数&#xff1a;1、to_string&#xff08;&#xff09;&#xff1a;2、stoi&#xff08;&#xff09;&#xff1a;3、substr&#xff08;&#xff09;&#xff1a;代码 题目 解析 首先…