千家信息网

Qt控制器管理的方法是什么

发表于:2025-12-02 作者:千家信息网编辑
千家信息网最后更新 2025年12月02日,这篇文章主要介绍"Qt控制器管理的方法是什么",在日常操作中,相信很多人在Qt控制器管理的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答"Qt控制器管理的方法是
千家信息网最后更新 2025年12月02日Qt控制器管理的方法是什么

这篇文章主要介绍"Qt控制器管理的方法是什么",在日常操作中,相信很多人在Qt控制器管理的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答"Qt控制器管理的方法是什么"的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

一、前言

控制器管理,主要就是对控制器进行添加删除和修改,其中包括编号、端口名称、控制器名称、控制器地址、控制器型号、探测器数量这几个字段,端口名称表示当前控制器所属哪个端口,一个系统中可以有好多个端口,一个端口可以对应多个控制器,一个控制器可以接多个探测器。探测器数量这个参数主要是方便下发数据的时候,提前知道要取多少个探测器的数据,组成对应的数据包下发下去,可以规避一下无效的数据,比如下面是16个探测器,就没有必要去取255个探测器的数据,取到了也没有意义,毕竟其余的全部是0,而且还增加了总线通信的负担,垃圾数据占用总线通道。

通信端口和控制器信息,是没有存储间隔的,要在最末端的节点探测器才需要有存储间隔,每个探测器节点都可以根据需要设定不一样的存储间隔,这个系统通信交互都是和通信端口下的控制器在通信,一旦控制器掉线了,则该控制器下面的所有探测器都掉线,一旦端口不通了,则该端口下的所有控制器都掉线,层层级联。

二、功能特点

  1. 采集数据端口,支持串口端口+网络端口,串口支持自由设置串口号+波特率,网络支持自由设置IP地址+通讯端口,每个端口支持采集周期,默认1秒钟一个地址,支持设置通讯超时次数,默认3次,支持最大重连时间,用于重新读取离线的设备。

  2. 控制器信息,能够添加控制器名称,选择控制器地址+控制器型号,设置该控制器下面的探测器数量。

  3. 探测器信息,能够添加位号,可自由选择探测器型号,气体种类,气体符号,高报值,低报值,缓冲值,清零值,是否启用,报警声音,背景地图,存储周期,数值换算小数点位数,报警延时时间,报警的类型(HH,LL,HL)等。

  4. 控制器型号+探测器型号+气体种类+气体符号,均可自由配置。

  5. 地图支持导入和删除,所有的探测器对应地图位置可自由拖动保存。

  6. 端口信息+控制器信息+探测器信息,支持导入导出+导出到excel+打印。

  7. 运行记录+报警记录+用户记录,支持多条件组合查询,比如时间段+控制器+探测器等,所有记录支持导出到excel+打印。

  8. 导出到excel的记录支持所有excel+wps等表格文件版本,不依赖excel等软件。

  9. 可删除指定时间范围内的数据,支持自动清理早期数据,设置最大保存记录数。

  10. 支持报警短信转发,支持多个接收手机号码,可设定发送间隔,比如即时发送或者6个小时发送一次所有的报警信息,短信内容过长,自动拆分多条短信。

  11. 支持报警邮件转发,支持多个接收邮箱,可设定发送间隔,比如即时发送或者6个小时发送一次所有的报警信息,支持附件发送。

  12. 高报颜色+低报颜色+正常颜色+0值颜色+曲线背景+曲线颜色等,都可以自由选择。

  13. 软件的中文标题+英文标题+logo路径+版权所有都可以自由设置。

  14. 提供开关设置开机运行+报警声音+自动登录+记住密码等。

  15. 报警声音可设置播放次数,界面提供17种皮肤文件选择。

  16. 支持云端数据同步,可设置云端数据库的信息,比如数据库名称,用户名+密码等。

  17. 支持网络转发和网络接收,网络接收开启后,软件从udp接收数据进行解析。网络转发支持多个目标IP,这样就实现了本地采集的软件,自由将数据转到客户端,随时查看探测器数据。

  18. 自动记住用户最后停留的界面+其他信息,重启后自动应用。

  19. 报警自动切换到对应的地图,探测器按钮闪烁。

  20. 双击探测器图标,可以进行回控。

  21. 支持用户权限管理,管理员+操作员两大类,用户登录+用户退出,可以记住密码和自动登录,超过三次报错提示并关闭程序。

  22. 支持四种监控模式,设备面板监控+地图监控+表格数据监控+曲线数据监控,可自由切换,四种同步应用。

  23. 支持报警继电器联动,一个位号可以跨串口联动多个模块和继电器号,支持多对多。

  24. 本地数据存储支持sqlite+mysql,支持远程数据同步到云端数据库。自动重连。

  25. 本地设备采集到的数据实时上传到云端,以便手机APP或者web等其他方式提取。

  26. 支持两种数据源,一种是串口和网络通过协议采集设备数据,一种是数据库采集。数据库采集模式可以作为通用的系统使用。

  27. 自带设备模拟工具,支持16个设备数据模拟,同时还带数据库数据模拟,以便在没有设备的时候测试数据。

  28. 默认通信协议采用modbus协议,后期增加mqtt等物联网协议的支持,做成通用系统。

  29. 支持所有windows操作系统+linux操作系统和其他操作系统。

三、效果图

四、核心代码

#include "frmconfigdevice.h"#include "ui_frmconfigdevice.h"#include "quiwidget.h"#include "dbhelper.h"#include "dbdelegate.h"#include "excelapi.h"#include "printapi.h"#include "api.h"frmConfigDevice::frmConfigDevice(QWidget *parent) : QWidget(parent), ui(new Ui::frmConfigDevice){    ui->setupUi(this);    this->initForm();    this->initData();    this->changeStyle();}frmConfigDevice::~frmConfigDevice(){    delete ui;}void frmConfigDevice::showEvent(QShowEvent *){    model->select();}void frmConfigDevice::initForm(){    API::initTableView(ui->tableView);    ui->widgetTop->setProperty("flag", "navbtn");    if (QUIHelper::deskWidth() < 1440) {        ui->labTip->setText("提示 → 改动后需重启应用");    } else {        ui->labTip->setText("提示 → 必须和现场控制器信息一致,改动后需重启应用");    }    }void frmConfigDevice::initData(){    model = new QSqlTableModel(this);    model->setTable("DeviceInfo");    model->setSort(0, Qt::AscendingOrder);    model->setEditStrategy(QSqlTableModel::OnManualSubmit);    model->select();    ui->tableView->setModel(model);    ui->tableView->setProperty("model", true);    columnNames.clear();    columnNames << "编号" << "端口名称" << "控制器名称" << "控制器地址" << "控制器型号" << "探测器数量";    columnWidths.clear();    columnWidths << 80 << 120 << 170 << 80 << 170 << 80;    for (int i = 0; i < columnNames.count(); i++) {        model->setHeaderData(i, Qt::Horizontal, columnNames.at(i));        ui->tableView->setColumnWidth(i, columnWidths.at(i));    }    //编号委托    QStringList deviceID;    for (int i = 1; i <= 255; i++) {        deviceID.append(QString::number(i));    }    DbDelegate *d_cbox_deviceID = new DbDelegate(this);    d_cbox_deviceID->setDelegateType("QComboBox");    d_cbox_deviceID->setDelegateValue(deviceID);    //ui->tableView->setItemDelegateForColumn(0, d_cbox_deviceID);    //通信端口委托    d_cbox_portName = new DbDelegate(this);    d_cbox_portName->setDelegateType("QComboBox");    ui->tableView->setItemDelegateForColumn(1, d_cbox_portName);    portNameChanged();    //设备地址委托    QStringList deviceAddr;    for (int i = 1; i <= 255; i++) {        deviceAddr.append(QString::number(i));    }    DbDelegate *d_cbox_deviceAddr = new DbDelegate(this);    d_cbox_deviceAddr->setDelegateType("QComboBox");    d_cbox_deviceAddr->setDelegateValue(deviceAddr);    ui->tableView->setItemDelegateForColumn(3, d_cbox_deviceAddr);    //控制器型号委托,关联控制器型号,下拉变化右侧值改动    d_cbox_deviceType = new DbDelegate(this);    connect(d_cbox_deviceType, SIGNAL(valueChanged(QString)), this, SLOT(typeValueChanged(QString)));    d_cbox_deviceType->setDelegateType("QComboBox");    ui->tableView->setItemDelegateForColumn(4, d_cbox_deviceType);    deviceTypeChanged();    //探测器数量委托    DbDelegate *d_txt_nodeNumber = new DbDelegate(this);    d_txt_nodeNumber->setDelegateType("QLineEdit");    if (App::WorkMode == 0) {        d_txt_nodeNumber->setDelegateEnable(false);    } else {        connect(d_txt_nodeNumber, SIGNAL(valueChanged(QString)), this, SLOT(numberValueChanged(QString)));    }    //connect(d_txt_nodeNumber, SIGNAL(valueChanged(QString)), this, SLOT(numberValueChanged(QString)));    ui->tableView->setItemDelegateForColumn(5, d_txt_nodeNumber);    //用来切换样式自动改变颜色    delegates << d_cbox_deviceID;    delegates << d_cbox_portName;    delegates << d_cbox_deviceAddr;    delegates << d_cbox_deviceType;    delegates << d_txt_nodeNumber;}void frmConfigDevice::changeStyle(){    foreach (DbDelegate *delegate, delegates) {        delegate->setTextColor(QUIConfig::TextColor);        delegate->setSelectBgColor(QUIConfig::NormalColorStart);    }}void frmConfigDevice::deviceTypeChanged(){    d_cbox_deviceType->setDelegateValue(DBData::TypeInfo_DeviceType);    }void frmConfigDevice::portNameChanged(){    d_cbox_portName->setDelegateValue(DBData::PortInfo_PortName);}void frmConfigDevice::typeValueChanged(const QString &value){    //自动设置最大探测器数量    int nodeNumber = 8;    int index = DBData::TypeInfo_DeviceType.indexOf(value);    if (index >= 0) {        nodeNumber = DBData::TypeInfo_NodeNumber.at(index);    }    int row = ui->tableView->currentIndex().row();    model->setData(model->index(row, 5), nodeNumber);}void frmConfigDevice::numberValueChanged(const QString &value){    //判断填写的最大值是否超过    int row = ui->tableView->currentIndex().row();    QString deviceType = model->index(row, 4).data().toString();    int index = DBData::TypeInfo_DeviceType.indexOf(deviceType);    if (index >= 0) {        int nodeNumber = DBData::TypeInfo_NodeNumber.at(index);        if (value.toInt() > nodeNumber) {            model->setData(model->index(row, 5), nodeNumber);            QUIHelper::showMessageBoxError(QString("当前控制器最大探测器数量为 %1 !\n请重新输入!").arg(nodeNumber));        }    }}void frmConfigDevice::on_btnAdd_clicked(){    int count = model->rowCount();    model->insertRow(count);    int deviceID = model->index(count - 1, 0).data().toInt() + 1;    QString portName = model->index(count - 1, 1).data().toString();    QString deviceName = model->index(count - 1, 2).data().toString();    int deviceAddr = model->index(count - 1, 3).data().toInt() + 1;    QString deviceType = model->index(count - 1, 4).data().toString();    int nodeNumber = model->index(count - 1, 5).data().toInt();    if (deviceName.endsWith("#调度室控制器")) {        deviceName = QString("%1#调度室控制器").arg(deviceAddr);    }    //设置第一条数据,在没有上一条参考数据时    if (count == 0) {        deviceID = 1;        portName = DBData::PortInfo_Count > 0 ? DBData::PortInfo_PortName.first() : "通信端口-1";        deviceName = "1#调度室控制器";        deviceAddr = 1;        deviceType = "FC-1003-8";        nodeNumber = 8;        if (App::WorkMode == 2) {            deviceName = "默认控制器";            deviceType = "WF-1000";            nodeNumber = 16;        }    }    //设置新增加的行默认值    model->setData(model->index(count, 0), deviceID);    model->setData(model->index(count, 1), portName);    model->setData(model->index(count, 2), deviceName);    model->setData(model->index(count, 3), deviceAddr);    model->setData(model->index(count, 4), deviceType);    model->setData(model->index(count, 5), nodeNumber);    ui->tableView->setCurrentIndex(model->index(count, 0));}void frmConfigDevice::on_btnSave_clicked(){    model->database().transaction();    if (model->submitAll()) {        model->database().commit();        DBHelper::loadDeviceInfo();        //先同步探测器表的控制器名称        emit deviceNameChanged();    } else {        model->database().rollback();        QUIHelper::showMessageBoxError("保存信息失败,信息不能为空,请重新填写!");    }}void frmConfigDevice::on_btnDelete_clicked(){    if (ui->tableView->currentIndex().row() < 0) {        QUIHelper::showMessageBoxError("请选择要删除的控制器!");        return;    }    if (QUIHelper::showMessageBoxQuestion("确定要删除该控制器吗? 控制器对应的所有探测器都会被删除!") == QMessageBox::Yes) {        int row = ui->tableView->currentIndex().row();        QString deviceName = model->index(row, 2).data().toString();        DBHelper::deleteNodeInfo(deviceName);        model->removeRow(row);        model->submitAll();        int count = model->rowCount();        ui->tableView->setCurrentIndex(model->index(count - 1, 0));    }}void frmConfigDevice::on_btnReturn_clicked(){    model->revertAll();}void frmConfigDevice::on_btnClear_clicked(){    int count = model->rowCount();    if (count <= 0) {        return;    }    if (QUIHelper::showMessageBoxQuestion("确定要清空所有控制器信息吗?") == QMessageBox::Yes) {        DBHelper::clearDeviceInfo();        model->select();    }}void frmConfigDevice::on_btnExcel_clicked(){    QString name = "控制器信息";    QString fileName = QString("%1_%2").arg(name).arg(QDateTime::currentDateTime().toString("yyyy-MM-dd-HH-mm-ss"));    QString file = API::getSaveFileNames(fileName, "Excel(*.xls)");    if (file.isEmpty()) {        return;    }    QString columns = "DeviceID,PortName,DeviceName,DeviceAddr,DeviceType,NodeNumber";    QString where = "order by DeviceID asc";    QStringList content = DBHelper::getContent("DeviceInfo", columns, where, "", ";");    int rowCount = content.count();    if (rowCount == 0) {        QUIHelper::showMessageBoxError("没有要处理的数据!");        return;    }    ExcelAPI::Instance()->saveExcel(file, name, name, "", columnNames, columnWidths, content);    QString msg = QString("导出%1到Excel").arg(name);    DBHelper::addUserLog("用户操作", msg);    if (QUIHelper::showMessageBoxQuestion(msg + "成功!确定现在就打开吗?") == QMessageBox::Yes) {        QString url = QString("file:///%1").arg(file);        QDesktopServices::openUrl(QUrl(url, QUrl::TolerantMode));    }}void frmConfigDevice::on_btnPrint_clicked(){    QString name = "控制器信息";    QString columns = "DeviceID,PortName,DeviceName,DeviceAddr,DeviceType,NodeNumber";    QString where = "order by DeviceID asc";    QStringList content = DBHelper::getContent("DeviceInfo", columns, where, "", ";");    int rowCount = content.count();    if (rowCount == 0) {        QUIHelper::showMessageBoxError("没有要处理的数据!");        return;    }    PrintAPI::Instance()->print(name, "", columnNames, columnWidths, content);    QString msg = QString("打印%1").arg(name);    DBHelper::addUserLog("用户操作", msg);}void frmConfigDevice::on_btnInput_clicked(){    QString fileName;    bool ok = DBHelper::inputData(columnNames.count(), App::FileFilter, "DeviceInfo", fileName, QUIHelper::appPath() + "/db", true);    if (!fileName.isEmpty()) {        DBHelper::addUserLog("用户操作", "导入控制器信息");        if (ok) {            QUIHelper::showMessageBoxInfo("导入控制器信息成功!", 3);            model->select();        } else {            QUIHelper::showMessageBoxError("导入控制器信息失败!", 3);        }    }}void frmConfigDevice::on_btnOutput_clicked(){    QString columns = "*";    QString where = "order by DeviceID asc";    QString title = columnNames.join(App::FileSpliter);    QStringList content = DBHelper::getContent("DeviceInfo", columns, where, title);    QString fileName;    bool ok = DBHelper::outputData("控制器信息", App::FileFilter, content, fileName, QUIHelper::appPath() + "/db");    if (!fileName.isEmpty()) {        DBHelper::addUserLog("用户操作", "导出控制器信息");        if (ok) {            QUIHelper::showMessageBoxInfo("导出控制器信息成功!", 3);        } else {            QUIHelper::showMessageBoxError("导出控制器信息失败!", 3);        }    }}

到此,关于"Qt控制器管理的方法是什么"的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注网站,小编会继续努力为大家带来更多实用的文章!

控制器 控制 数据 支持 探测器 探测 信息 端口 报警 用户 自由 名称 型号 设备 通信 管理 多个 数量 系统 网络 数据库的安全要保护哪些东西 数据库安全各自的含义是什么 生产安全数据库录入 数据库的安全性及管理 数据库安全策略包含哪些 海淀数据库安全审计系统 建立农村房屋安全信息数据库 易用的数据库客户端支持安全管理 连接数据库失败ssl安全错误 数据库的锁怎样保障安全 移动通信机房网络技术知识 cmd打开数据库服务器 主干网络技术的选择 网络技术服务操作 万博网络安全工程师 电脑打印机服务器笔记本回收 t6的现存量数据库是哪个 网络安全记录软件 软件开发工程师证书有没有 沭阳多功能网络技术市场 兰州软件开发待遇怎么样 红古区人民网络安全网 网络安全目前存在哪些隐患 公司软件开发手册 网络安全领域规划 网络安全与执法专业具体课程 重庆智慧城市管理平台软件开发 sql文件怎样导入数据库中 艾尔登法环服务器登不上 数据库的安全漏洞不属于软件漏洞 网络安全美篇宣传 互联网科技助力厅堂转型 字节字节网络技术公司 我的世界服务器锁ip怎么开小号 希望之村服务器满进不去怎么办 网络安全案件分析表 梅县定制软件开发 qq安全中心提醒服务器 华为腾讯阿里的服务器都在哪里 逆水寒两周年服务器总评分第一
0