【开源、应用】QT—TCP网络上位机的设计

news2025/1/10 3:07:25

本文设计一个终端控制的上位机软件(如“设计目标”下图所示),可以和STM32、Adruino等通信实现无线局域网控制系统。


本文的通信内容和图表内容可以参考作者之前的文章

STM32+ESP8266连接电脑Qt网络上位机——QT篇https://blog.csdn.net/qq_53734051/article/details/126706759?spm=1001.2014.3001.5501

QT—Qcharts绘制实时曲线https://blog.csdn.net/qq_53734051/article/details/126872728?spm=1001.2014.3001.5501


目录

一、设计目标

二、设计原理

三、通信部分

四、图表部分

五、数据刷新和UI设计


一、设计目标

        利用组件容器等,结合通信,发送相应格式的文本,在主控解析后做出相应的回应;同样在数据上报也需要主控发送指定的格式文本,上位机才能正确的解析。下方评论+邮箱发送源码或Git自行下载:https://gitee.com/guo-lingran/tcp-sql-network-host-computer

百度网盘下载连接:https://pan.baidu.com/s/1AROPIT3big1puhzCKyRTSQ  提取码:pumk

哔哩哔哩视频演示链接 —>:【开源、应用】QT—TCP+Sql网络上位机的设计_哔哩哔哩_bilibili

 

二、设计原理

 该软件主要分为以下部分:

        通信部分QTcpServer、QTcpSocket

        图表部分QChartView、QLineSeries、QValueAxis、QDateTimeAxis

        数据刷新QTimer

        UI槽函数设计QToolBar、QLabel、QLineEdit、QPushButton、QSlider、QCheckBox等)

三、通信部分

         之前的文章已经写过TCP通信实现过程以及部分代码,所以这里不再详写 —> 点击跳转STM32+ESP8266连接电脑Qt网络上位机—QT篇https://blog.csdn.net/qq_53734051/article/details/126706759?spm=1001.2014.3001.5501

这里有点小的变化,在工具栏里可以再调出一个窗口,可供调试使用

调用该窗口

        头文件需要定义一下该类 :

private:
Deb *deb = new Deb;

调出窗口即可

//调出调试窗口
void MainWindow::on_debb_triggered()
{
    deb->show();
}

 关闭窗口

void MainWindow::on_exit_triggered()
{
    this->close();
}

容器接收

        当软件接收数据后,一部分传入界面二(调试界面),供调试使用,另一部分进行解析,并进入图表或文本(下面再讲解)。如下代码->

//收到的数据放入接受框   解析
void MainWindow::readyRead_Slot(){
    QByteArray receiveDate;
    QTextCodec *tc = QTextCodec::codecForName("GBK"); 

    while(!tcpSocket->atEnd()){
        receiveDate = tcpSocket->readAll();
    }

    if (!receiveDate.isEmpty())
    {
        QString strBuf=tc->toUnicode(receiveDate);       
        //传入界面二数据
        deb->DisplayData(strBuf);
        //解析到可视化曲线图和文本
        BackDataParsing(strBuf);
    }
    receiveDate.clear();
}

界面二中的数据显示

        如下图

void Deb::DisplayData(QString qstring){
    ui->rec_edi->appendPlainText(qstring);
}

容器发送

        在界面二直接进行数据发送后,会引起阻塞,所以为了保证发送成功,这里在界面一使用定时器来判断是否有数据发送,这样两个界面不会有冲突。

/*构造函数*/
timer_send = new QTimer(this);
timer_send->start(20);
connect(timer_send,SIGNAL(timeout()),this,SLOT(timer_send_Slot()));
/*END*/

void MainWindow::timer_send_Slot(){
    if(flag_Send==1){
        flag_Send=0;
      tcpSocket->write(strbuf.toLocal8Bit().data());
    }
}

四、图表部分

之前的文章也介绍过怎么绘制实时曲线,这里不再详写


//创建chart
void MainWindow::creatChart()
{
    QChart *qchart = new QChart();
    //把chart放到容器里
    ui->graphicsView->setChart(qchart);
    ui->graphicsView->setRenderHint(QPainter::Antialiasing); //设置抗锯齿

    //创建两条线
    QLineSeries *series0 = new QLineSeries;
    QLineSeries *series1 = new QLineSeries;
    QLineSeries *series2 = new QLineSeries;

    //设置名字
    series0->setName("温度");
    series1->setName("湿度");
    series2->setName("光强");

    //把线条放到chart里
    qchart->addSeries(series0);
    qchart->addSeries(series1);
    qchart->addSeries(series2);

    //创建x 坐标
    QDateTimeAxis *axisX = new QDateTimeAxis;

    //格式
    axisX->setFormat("hh:mm:ss");
    //设置竖条数量
    axisX->setTickCount(5);

    //设置坐标名称
    axisX->setTitleText("time(sec)");

    qchart->setAxisX(axisX,series0);
    qchart->setAxisX(axisX,series1);
    qchart->setAxisX(axisX,series2);

    //创建y坐标
    QValueAxis  *axisY = new QValueAxis;
    axisY->setRange(0,100);
    axisY->setTickCount(5);

    qchart->setAxisY(axisY,series0);
    qchart->setAxisY(axisY,series1);
    qchart->setAxisY(axisY,series2);

    qchart->setDropShadowEnabled(true);

    //初始化坐标
         //设置最大值坐标值 系统时间当前时间
    qchart->axisX()->setMin(QDateTime::currentDateTime().addSecs(0));
         //设置最大值坐标值 系统时间后5*30秒
    qchart->axisX()->setMax(QDateTime::currentDateTime().addSecs(5*30));

}

QT—Qcharts绘制实时曲线https://blog.csdn.net/qq_53734051/article/details/126872728?spm=1001.2014.3001.5501        发送格式:Params{temp:39.3;humi:82.9;light:69.3;soil:38.3;mq2:22.2;rain:57.3;}

        当接收到数据后,进入BackDataParsing函数进行文本解析,在解析文本时使用mid函数取指定长度的字符串,.indexOf定位关键词,返回关键词的位置(int)。

解析原理:

.mid有两个参数,第一个为起始位置(int),第二个为取值长度。

起始位置:

        首先Params为起始,利用.indexof定位到第一个关键词的位置,并加上该关键词的长度,即取到:后面;

取值长度:

        再利用.indexof定位到第二个关键词的位置,并减去第一个关键词的位置和长度,以此算出该有效数据的长度。

void MainWindow::BackDataParsing(QString strBuf){

    //查找是否为参数;  -1表示没有该子串
 if(strBuf.startsWith("Params")){

        //表一数据
      QString str = strBuf.mid(strBuf.indexOf("temp:")+((QString)"temp:").length(),strBuf.indexOf("humi:")-strBuf.indexOf("temp:")-((QString)"temp:").length()-1);
      tcpSocket->write(str.toUtf8());
      tcpSocket->write("->");

      QString st2 = strBuf.mid(strBuf.indexOf("humi:")+((QString)"humi:").length(),strBuf.indexOf("light:")-strBuf.indexOf("humi:")-((QString)"humi:").length()-1);
      tcpSocket->write(st2.toUtf8());
      tcpSocket->write("->");

      QString st3 = strBuf.mid(strBuf.indexOf("light:")+((QString)"light:").length(),strBuf.indexOf("soil:")-strBuf.indexOf("light:")-((QString)"light:").length()-1);
      tcpSocket->write(st3.toUtf8());
      tcpSocket->write("->");

      //表二数据
      QString st4 = strBuf.mid(strBuf.indexOf("soil:")+((QString)"soil:").length(),strBuf.indexOf("mq2:")-strBuf.indexOf("soil:")-((QString)"soil:").length()-1);
      tcpSocket->write(st4.toUtf8());
      tcpSocket->write("->");

      QString st5 = strBuf.mid(strBuf.indexOf("mq2:")+((QString)"mq2:").length(),strBuf.indexOf("rain:")-strBuf.indexOf("mq2:")-((QString)"mq2:").length()-1);
      tcpSocket->write(st5.toUtf8());
      tcpSocket->write("->");

      QString st6 = strBuf.mid(strBuf.indexOf("rain:")+((QString)"rain:").length(),strBuf.indexOf("}")-strBuf.indexOf("rain:")-((QString)"rain:").length()-1);
      tcpSocket->write(st6.toUtf8());

       temp_data = str.toFloat();
       humi_data = st2.toFloat();
       light_data = st3.toFloat();

       soil_data = st4.toFloat();
       mq2_data = st5.toFloat();
       rain_data = st6.toFloat();

       ToUpdata_Lab(str,st2,st3,st4,st5,st6);

   }
}

解析完后更显到表里和显示区域里 (ToUpdata_Lab)

文本区域显示

//文本显示
void MainWindow::ToUpdata_Lab(QString Stemp,QString Shumi,QString Slight,QString Ssoil,QString Smq2,QString Srain){
    ui->temp_la->setText(Stemp+"%");
    ui->humi_la->setText(Shumi+"%");
    ui->light_la->setText(Slight+"%");

    ui->soil_la->setText(Ssoil+"%");
    ui->mq2_la->setText(Smq2+"%");
    ui->rain_la->setText(Srain+"%");

}

表格刷新

        动态变化详见上篇文章

//表一刷新
void MainWindow::DisplayChart1(){
    //获取当前时间
    QDateTime currentTime = QDateTime::currentDateTime();

    //获取初始化的qchart
    QChart *qchart =(QChart *)ui->graphicsView->chart();
    //获取初始化的series;
    QLineSeries *series0 = (QLineSeries *)ui->graphicsView->chart()->series().at(0);
    QLineSeries *series1 = (QLineSeries *)ui->graphicsView->chart()->series().at(1);
    QLineSeries *series2 = (QLineSeries *)ui->graphicsView->chart()->series().at(2);

    series0->append(currentTime.toMSecsSinceEpoch(),temp_data);
    series1->append(currentTime.toMSecsSinceEpoch(),humi_data);
    series2->append(currentTime.toMSecsSinceEpoch(),light_data);

    qchart->axisX()->setMin(QDateTime::currentDateTime().addSecs(-5*30));
    qchart->axisX()->setMax(QDateTime::currentDateTime().addSecs(5*30));
}

 五、数据刷新和UI设计

数据刷新部分包括表格的动态变化、时间显示和继电器自动关停(蓄水ui的动态变化


//时间刷新
void MainWindow::ReData_Slot(){
    static int timer=0;

    DisplayChart1();
    DisplayChart2();


    //当前时间
    ui->time_l->setText(QTime::currentTime().toString("hh:mm:ss"));

    //继电器控制
    if(relaySw){
        timer++;
        // boVal 设定值  timer 当前值
        int boVal = ui->spinBox->value();

        ui->progressBar->setValue((timer*100)/boVal);

            //超时后自动关闭
        if(timer>=boVal){
            timer=0;
            relaySw=false;
            ui->relay->setIcon(QIcon(":/relay_off.png"));
            tcpSocket->write("relay_off");
            ui->progressBar->setValue(100);
        }
      }
    else
    {    
        timer=0;
    }
}

阈值设定

        采用QCheckBox多选框完成一个阈值的设定,需要设定哪一个阈值勾选即可,然后设定。

发送格式: 状态+soil:12;状态+rain:20;状态+temp:5;状态+light:6

若设定为以下的阈值,将会发送这样的数据:enable soil:12;enable rain:20;enable temp:5;enable light:6   (如下图)

 enable soil:30;disable rain:52;disable temp:6;enable light:62

 

 最后用于单片机的解析,并作出相应的变化。

槽函数:

void MainWindow::on_set_yu_bt_clicked()
{
    QString sendThrshold;

    sendThrshold = EnsoilHumi + " " + "soil:"+ ui->soil_yu_la->text()+";"+
                   Enrain     + " " + "rain:"+ ui->rain_yu_la->text()+";"+
                   Entemp     + " " + "temp:"+ ui->temp_yu_la->text()+";"+
                   Enlight    + " " + "light:"+ui->light_yu_la->text();

    tcpSocket->write(sendThrshold.toLocal8Bit());

}
//复选框  2选中 0未选中
void MainWindow::on_checkBox_stateChanged(int arg1)
{
    qDebug()<<arg1;
    if(arg1==2)
        EnsoilHumi="enable";
    else
        EnsoilHumi="disable";
}
void MainWindow::on_checkBox_2_stateChanged(int arg1)
{
    qDebug()<<arg1;

    if(arg1==2)
         Enrain="enable";
    else
         Enrain="disable";
}
void MainWindow::on_checkBox_3_stateChanged(int arg1)
{
     qDebug()<<arg1;

     if(arg1==2)
          Entemp="enable";
     else
          Entemp="disable";
}
void MainWindow::on_checkBox_4_stateChanged(int arg1)
{

    if(arg1==2)
         Enlight="enable";
    else
         Enlight="disable";
}

光强设定 

        和上面一样,只不过用到了一个QSlider的类,设定26%后,发送格式:Pwm:+值,同样的单片机解析后做出相对的改变。


//滑动改变
void MainWindow::on_horizontalSlider_valueChanged(int value)
{
     ui->setlightNumla->setNum(value);
     light_pwm = value;
}

//光强控制
void MainWindow::on_set_light_bt_clicked()
{
   tcpSocket->write(("Pwm:"+QString::number(light_pwm)).toLocal8Bit());
}

 QToolBar工具栏

        工具栏包括网络的开启、LED开关、继电器开关、模式切换、调试窗口和退出软件。原理都一样,代码部分不再列出。

 蓄水状态

        当打开继电器开关后,会有一个倒计时自动关闭的设计,这里的时间可以设定 (如下图)。达到100%后自动关闭,并发送命令:relay_off 。

        这里使用了QProgressBar和QSpinBox的类,在定时器里1s发生改变

               

//继电器控制
if(relaySw){
    timer++;
    // boVal 设定值  timer 当前值
    int boVal = ui->spinBox->value();
    //进度条设定
    ui->progressBar->setValue((timer*100)/boVal);
    //超时后自动关闭
    if(timer>=boVal){
        timer=0;
        relaySw=false;
        ui->relay->setIcon(QIcon(":/relay_off.png"));
        tcpSocket->write("relay_off");
        ui->progressBar->setValue(100);
       }
   }

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【STM32G431RBTx】备战蓝桥杯嵌入式→决赛试题→第六届

文章目录 前言一、题目二、模块初始化三、代码实现interrupt.h:interrupt.c:main.h:main.c: 四、完成效果五、总结 前言 无 一、题目 二、模块初始化 1.LCD这里不用配置&#xff0c;直接使用提供的资源包就行 2.ADC:开启ADCsingle-ended 3.LED:开启PC8-15,PD2输出模式就行了…

【JVM001】宋红康JVM字节码举例

宋红康JVM字节码举例 1 Integer package jvmT; public class IntegerTest {public static void main(String[] args) {Integer i 5;int y 5;System.out.println(iy); //trueInteger i6 5;Integer y6 5;System.out.println(i6y6);//trueInteger i5 128;Integer y5 128;System.…

SpringBoot中使用lombok

1.添加依赖 在项目的根目录中找到pom.xml&#xff0c;在dependencies下复制这段代码 <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifac…

解决Python爬虫中selenium模块中的find_element_by_id方法无法使用

如有错误&#xff0c;敬请谅解&#xff01; 此文章仅为本人学习笔记&#xff0c;仅供参考&#xff0c;如有冒犯&#xff0c;请联系作者删除&#xff01;&#xff01; 我们在学习selenium模块的时候&#xff0c;经常会用到 browser.find_element_by_id命令&#xff0c;但随着se…

代码随想录算法训练营第四十六天 | bool的背包题,细节多

139.单词拆分 文档讲解&#xff1a;代码随想录 (programmercarl.com) 视频讲解&#xff1a;动态规划之完全背包&#xff0c;你的背包如何装满&#xff1f;| LeetCode&#xff1a;139.单词拆分_哔哩哔哩_bilibili 状态&#xff1a;不会做&#xff0c;不知道怎么把bool类型与背包…

推荐系统系列之推荐系统概览(下)

在推荐系统概览的第一讲中&#xff0c;我们介绍了推荐系统的常见概念&#xff0c;常用的评价指标以及首页推荐场景的通用召回策略。本文我们将继续介绍推荐系统概览的其余内容&#xff0c;包括详情页推荐场景中的通用召回策略&#xff0c;排序阶段常用的排序模型&#xff0c;推…

软件测试实验:loadrunner的高级使用

目录 前言实验目的实验内容实验要求实验过程loadrunner中插入事务与集合点loadrunner中插入检查点loadrunner中参数化-table分析报告功能loadrunner手动设置场景loadrunner监视图标 总结 前言 本实验主要介绍了loadrunner这一强大的性能测试工具的高级使用方法&#xff0c;包括…

python实现九宫格的车辆路径轨迹上位机界面

实验环境&#xff1a;wxFormBuilder v3.5 python3.7.5 MC9S12G128开发板 基本功能&#xff1a;控制开发板上的按键&#xff0c;模拟车辆移动的上下左右四个方位&#xff0c;通过can通信告诉上位机界面&#xff0c;车辆轨迹的移动方位&#xff1b; 1. python重新封装control…

技巧:jetbrain全家桶系列如何撤销已经提交本地仓库但还没push的commit

目录 1. 哎呀&#xff0c;不小心把不能提交的“机密”加入commit了2. 使用reset来修复的话要注意有坑&#xff0c;选Soft和Mixed&#xff0c;千万别选Hard和Keep3. 使用revert&#xff0c;只能修修补补&#xff0c;但commit还在&#xff0c;当然有好处是会留下使用痕迹&#xf…

异常处理机制

编程错误 编写程序时遇到的错误可大致分为 2 类&#xff0c;分别为语法错误和运行时错误。 语法错误 语法错误&#xff0c;也就是解析代码时出现的错误。当代码不符合Python语法规则时&#xff0c;Python解释器在解析时就会报出SyntaxError语法错误&#xff0c;与此同时还会…

服务(第二十六篇)redis的主从复制、哨兵、集群

主从复制&#xff1a; 主从复制&#xff0c;是指将一台Redis服务器的数据&#xff0c;复制到其他的Redis服务器。前者称为主节点(Master)&#xff0c;后者称为从节点(Slave)&#xff1b;数据的复制是单向的&#xff0c;只能由主节点到从节点。 原理&#xff1a; 主从关系确定…

[算法前沿]--009-HuggingFace介绍(大语言模型底座)

基础介绍 HuggingFace 是一家专注于自然语言处理(NLP)、人工智能和分布式系统的创业公司,创立于2016年。最早是主营业务是做闲聊机器人,2018年 Bert 发布之后,他们贡献了一个基于 Pytorch 的 Bert 预训练模型,即 pytorch-pretrained-bert,大受欢迎,进而将重心转向维护…

PoseiSwap以2500万美元估值,再获新一轮融资

近日&#xff0c;Nautilus Chain 上的首个 DEX PoseiSwap 宣布&#xff0c;其目前已经以 2500 万美元的估值&#xff0c;从 Gate Labs、Emurgo Ventures、Republic以及Cipholio Ventures 等行业顶级投资机构中&#xff0c;获得了新一轮的融资&#xff0c;不过目前该融资的具体数…

asp.net网上捐赠系统

一该源码功能十分的全面&#xff0c;具体介绍如下&#xff1a; 本版本软件主要完成三个功能&#xff1a; 1、建立网上捐赠功能 实现网上捐赠程序自动化&#xff0c;智能化&#xff0c;在捐赠者与受捐者填写各种资料后&#xff0c;自动保存方便以后调阅查询&#xff0c…

Java【网络编程1】详解DatagramSocket和DatagramPacket类, 逐行代码解析如何服务器客户端通信(附代码)

文章目录 前言一、认识 Socket(套接字), TCP 协议和 UDP 协议1, 什么是 Socket(套接字)2, 浅谈 TCP 协议和 UDP 协议的区别和特点 二、基于 UDP 协议的 Socket API1, DatagramSocket 类2, DatagramPacket 类 三、逐行代码解析网络编程1, 逐行解析客户端1.1, 核心成员方法 start…

【C++】-类和对象完结(内部类、匿名对象以及编译器的优化的讲解)(下)

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树 ❤️‍&#x1fa79;作者宣言&#xff1a;认真写好每一篇博客 &#x1f4a8;作者gitee:gitee &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 如 果 你 喜 欢 作 者 的 文 章 &#xff0c;就 给 作 者 点 …

m1安装svn

背景&#xff1a;电脑是mac m2&#xff0c;好多软件都不太兼容&#xff0c;安装软件成了一个问题&#xff0c;想着装一个SVN&#xff0c;跟大家一起协同开发&#xff0c;这下可麻烦死了&#xff0c;&#x1f604;&#xff0c;终于弄明白用brew命令了&#xff0c;然后就用brew命…

Metasploitable2靶机渗透学习

目录 一、介绍 二、环境 三、渗透攻击 1.前期渗透 1.1主机发现 1.2.端口扫描 1.3.测试漏洞 2.弱密码漏洞 2.1系统弱密码登录&#xff08;telnet &#xff1a;23端口&#xff09; 2.2 MySQL弱密码登录&#xff08;端口&#xff1a;3306&#xff09; 2.3 PostgreSQL弱…

K8s全套快速入门

K8s快速入门 1 介绍 google开源的容器化管理工具机器数量十几台、上百台时&#xff0c;就可以考虑使用k8s高可用、自动容灾恢复、灰度更新、一键回滚历史版本、方便伸缩扩展等 k8s集群架构&#xff1a; 通常&#xff1a;一主多从 master&#xff1a;主节点&#xff0c;控制平台…

LeetCode 栈和队列OJ题目分享

目录 有效的括号&#xff08;括号匹配&#xff09;用栈实现队列用队列实现栈设计循环队列 有效的括号&#xff08;括号匹配&#xff09; 链接: link 题目描述&#xff1a; 题目思路&#xff1a; 1、如果是左括号“&#xff08; { [ ”就入栈 2、如果是右括号“&#xff09; }…