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

【Qt】Qt + Modbus 服务端学习笔记

《Qt + Modbus 服务端学习笔记》

1.因为项目的需要,要写一个modbus通信,csdn上感觉有些回答,代码是人工智能生成的,有些细节不对。我这个经过实测,是可以直接用的。

首先要包含Qt 的相关模块

在这里插入图片描述

Qt Modbus 模块主要包含以下几类关键组件:

  • 设备类:如 QModbusTcpServerQModbusTcpClient,分别用于创建 Modbus TCP 服务器和客户端,提供了建立连接、断开连接等基础操作接口。

  • 数据单元类:例如 QModbusDataUnit,用于表示 Modbus 协议中的数据单元,可方便地操作和管理 Modbus 寄存器中的数据。

  • 协议数据单元类:像 QModbusPdu,它表示 Modbus 协议数据单元,用于处理 Modbus 请求和响应的协议层数据。

  • 初始化:在 ModeBusServer 类的构造函数里,初始化 QModbusTcpServer 和定时器,同时建立信号与槽的连接,以处理定时器超时、数据写入、状态改变等事件。

  • 启动与停止

    • startServer 函数会对输入参数进行检查,设定 Modbus 数据单元映射和连接参数,尝试连接设备,若成功就启动数据更新定时器。
    • stopServer 函数则停止定时器,断开服务器连接。
  • 数据更新:借助定时器,定时调用 updateData 函数。此函数从 SystemInfoCollector 获取系统信息,将其写入 Modbus 数据单元,再设置到服务器里。

  • 数据处理

    • printUpdateData 函数用于打印更新后的数据和解析出的系统信息。
    • handleRequest 函数处理客户端请求,记录请求信息。
    • onDataWritten 函数处理数据写入事件。

运行结果

在这里插入图片描述

#ifndef MODEBUSSERVER_H
#define MODEBUSSERVER_H

#include <QObject>
#include <QModbusTcpServer>
#include <QModbusDataUnit>
#include <QTimer>
#include <QMap>
#include <memory> // 包含 std::unique_ptr 所在的头文件
#include <iostream>
#include <QDateTime>

#include "SystemInfoCollector.h"
//解决中文乱码
#pragma execution_character_set("utf-8")



class ModeBusServer : public QObject
{
    Q_OBJECT
public:
    explicit ModeBusServer(QObject* parent = nullptr);
    ~ModeBusServer();

    void startServer(quint16 deviceID, const QString& ipAddress, quint16 port);
    void stopServer();

    void printUpdateData(QModbusDataUnit& unit);

signals:
    void statusUpdated(const QString& message);
    void dataUpdated(const QModbusDataUnit& data);
    void requestReceived(const QModbusPdu& request, QModbusDataUnit::RegisterType table, int address, int size);


private slots:
    void updateData();
    void writeSystemInfoToModbus(QModbusDataUnit& unit, SystemInfo& info);
    void handleRequest(const QModbusPdu& request, QModbusDataUnit::RegisterType table, int address, int size);
    void sendDataPeriodically();

    void onDataWritten(QModbusDataUnit::RegisterType table, int address, int size);
private:
    std::unique_ptr<QModbusTcpServer> m_modbusServer;
    std::unique_ptr<QTimer> m_dataUpdateTimer;
    std::unique_ptr<QTimer> m_sendDataTimer;
    QMap<int, quint16> m_dataCache; // 数据缓存
    quint16 m_holdingRegistersSize = 100;
    int m_deviceID;
    int m_valueIndex;
    SystemInfoCollector m_infoCollector;
};

#endif // MODEBUSSERVER_H


#include "modeBusServer.h"
#include <QDebug>
ModeBusServer::ModeBusServer(QObject* parent)
	: QObject(parent),
	m_modbusServer(std::make_unique<QModbusTcpServer>(this)),
	m_dataUpdateTimer(std::make_unique<QTimer>(this)),
	m_sendDataTimer(std::make_unique<QTimer>(this))
{

	connect(m_dataUpdateTimer.get(), &QTimer::timeout, this, &ModeBusServer::updateData);
	connect(this, &ModeBusServer::statusUpdated, this, [&](const QString& statusUpdateStr) {
		qDebug() << statusUpdateStr;
		});
	// 修改连接,使用新的槽函数
	connect(m_modbusServer.get(), &QModbusTcpServer::dataWritten, this, &ModeBusServer::onDataWritten);

	connect(m_modbusServer.get(), &QModbusTcpServer::stateChanged, this, [this](int state) {
		if (state == QModbusDevice::ConnectedState) {
			qDebug() << "Server connected";
		}
		else if (state == QModbusDevice::UnconnectedState) {
			qDebug() << "Server disconnected";
		}
		});
}
ModeBusServer::~ModeBusServer()
{
	stopServer();
}
void ModeBusServer::startServer(quint16 deviceID, const QString& ipAddress, quint16 port)
{
	if (port == 0 || deviceID == 0 || ipAddress.isEmpty()) {
		emit statusUpdated("Invalid parameters.");
		return;
	}
	m_deviceID = deviceID;
	QModbusDataUnitMap reg;
	reg.insert(QModbusDataUnit::HoldingRegisters, { QModbusDataUnit::HoldingRegisters, 0, m_holdingRegistersSize });
	m_modbusServer->setMap(reg);

	m_modbusServer->setConnectionParameter(QModbusDevice::NetworkAddressParameter, ipAddress);
	m_modbusServer->setConnectionParameter(QModbusDevice::NetworkPortParameter, port);
	m_modbusServer->setServerAddress(deviceID);

	if (m_modbusServer->connectDevice()) {
		emit statusUpdated("Server started on " + ipAddress + ":" + QString::number(port));
		m_dataUpdateTimer->start(500);

	}
	else {
		emit statusUpdated("Server start failed: " + m_modbusServer->errorString());
	}
}
void ModeBusServer::stopServer()
{
	m_dataUpdateTimer->stop();
	m_modbusServer->disconnectDevice();
	emit statusUpdated("Server stopped.");
}
void ModeBusServer::printUpdateData(QModbusDataUnit& unit)
{

		for (int i = 0; i < unit.valueCount(); ++i) {

			m_dataCache[i] = unit.value(i); // 更新缓存
		}

		qDebug() << QString("%1号设备").arg(m_deviceID) << "寄存器数据:" << unit.values();
		SystemInfo info1 = m_infoCollector.parseSystemInfo(unit.values());
		qDebug() << "Memory Usage:" << info1.memoryUsage << "%";
		qDebug() << "CPU Usage:" << info1.cpuUsage << "%";
		qDebug() << "Boot Time:" << info1.bootTime.toString(Qt::ISODate);
		qDebug() << "Up Time:" << info1.upTime << "seconds";

}
// 更新数据
void ModeBusServer::updateData()
{
	m_valueIndex = 0;
	QModbusDataUnit unit(QModbusDataUnit::HoldingRegisters, 0, 10);

	SystemInfo info;
	// 将系统信息写入 Modbus 数据单元
	writeSystemInfoToModbus(unit, info);
	bool  isSetDataSuccess = m_modbusServer->setData(unit);


	//设置数据
	if (isSetDataSuccess)
	{
		m_dataCache.clear(); // 清空缓存
		printUpdateData(unit);
		emit dataUpdated(unit);
	}
	else
	{
		qDebug() << "Failed to update data: " << m_modbusServer->errorString();
	}


}
// 将 SystemInfo 数据写入 QModbusDataUnit
void ModeBusServer::writeSystemInfoToModbus(QModbusDataUnit& unit, SystemInfo& info) {

	info = m_infoCollector.getSystemInfo();

	// 写入内存占用率
	quint16 memoryUsageInt = static_cast<quint16>(info.memoryUsage * 100); // 转换为整数
	unit.setValue(m_valueIndex++, memoryUsageInt & 0xFFFF);

	// 写入 CPU 占用率
	quint16 cpuUsageInt = static_cast<quint16>(info.cpuUsage * 100); // 转换为整数
	unit.setValue(m_valueIndex++, cpuUsageInt & 0xFFFF);

	// 写入开机时间(时间戳)
	qint64 bootTimestamp = info.bootTime.toSecsSinceEpoch();
	unit.setValue(m_valueIndex++, static_cast<quint16>(bootTimestamp & 0xFFFF));
	unit.setValue(m_valueIndex++, static_cast<quint16>((bootTimestamp >> 16) & 0xFFFF));

	// 写入运行时间
	unit.setValue(m_valueIndex++, static_cast<quint16>(info.upTime & 0xFFFF));
	unit.setValue(m_valueIndex++, static_cast<quint16>((info.upTime >> 16) & 0xFFFF));

}
void ModeBusServer::handleRequest(const QModbusPdu& request, QModbusDataUnit::RegisterType table, int address, int size)
{
	emit requestReceived(request, table, address, size);
	qDebug() << "Request received: Function Code" << request.functionCode() << ", Type" << table << ", Address" << address << ", Size" << size;
}
void ModeBusServer::sendDataPeriodically()
{
	qDebug() << "Periodic Data Send:";
	for (auto it = m_dataCache.begin(); it != m_dataCache.end(); ++it) {
		qDebug() << "Address" << it.key() << "@" << it.value();
	}
}
// 实现新增的槽函数
void ModeBusServer::onDataWritten(QModbusDataUnit::RegisterType table, int address, int size)
{
	// 处理数据写入事件,可根据需要扩展
	qDebug() << "写入数据类型" << table << "从地址" << address << "开始。" << "数据长度:" << size;

}
	}
}
// 实现新增的槽函数
void ModeBusServer::onDataWritten(QModbusDataUnit::RegisterType table, int address, int size)
{
	// 处理数据写入事件,可根据需要扩展
	qDebug() << "写入数据类型" << table << "从地址" << address << "开始。" << "数据长度:" << size;

}

http://www.kler.cn/a/596037.html

相关文章:

  • 大腾智能受邀出席2025华为云城市峰会暨东莞市人工智能大模型中心开服活动
  • 快速查询手机是否处于联网状态?
  • Redis的大Key问题如何解决?
  • 什么是 DAO?
  • MinIO Docker
  • 数据可信安全流通实战,隐语开源社区Meetup武汉站开放报名
  • FPGA中级项目4——DDS实现
  • 工业数据驱动智能维护的深度调研报告
  • 基于Python的垃圾短信分类
  • 【AI论文】DropletVideo:一种用于探索整体时空一致视频生成的数据集与方法
  • 三层交换实验:实现不同 VLAN 间通信的详细探究
  • 气候预测新模式:助力行业迎接未来挑战
  • 音频录制小妙招-自制工具-借助浏览器录一段单声道16000采样率wav格式音频
  • RAG 架构地基工程-Retrieval 模块的系统设计分享
  • 论文笔记(七十三)Gemini Robotics: Bringing AI into the Physical World
  • HarmonyOS next性能优化:多维度策略与实战案例
  • 同旺科技USB to I2C 适配器 ---- 扫描I2C总线上的从机地址
  • Function Calling的核心机制与挑战
  • Python接口自动化浅析unittest单元测试原理
  • GEO与AISEO全面解析