8.21 QT

2024-08-22 18:20
文章标签 qt 8.21

本文主要是介绍8.21 QT,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.思维导图

2.

服务器端

头文件

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QTcpServer>//服务器类
#include <QMessageBox>
#include <QDebug>
#include <QList>
#include <QTcpSocket>QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACEclass Widget : public QWidget
{Q_OBJECTpublic:Widget(QWidget *parent = nullptr);~Widget();public slots:void newConnection_slot();void readyRead_slot();private slots:void on_start_btn_clicked();private:Ui::Widget *ui;//实例化一个服务器只针对象QTcpServer *server;QList <QTcpSocket *> socketList;
};
#endif // WIDGET_H

源文件

#include "widget.h"
#include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget), server(new QTcpServer(this))//给服务器指针对象实例空间
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}//newConnection信号对应的槽函数实现
void Widget::newConnection_slot()
{qDebug() << "有新的客户端连接。。";//使用nextPaddingConnection() 获取最新连接客户端的套接字QTcpSocket *s = server->nextPendingConnection();//将客户端放入容器中socketList.push_back(s);//程序运行至此,说明客户端和服务器成功建立了连接,//就可以将该信号连接到自定义的槽函数中,读取发来的数据connect(s,&QTcpSocket::readyRead,this,&Widget::readyRead_slot);
}//readyRead()信号对应的槽函数
void Widget::readyRead_slot()
{//读取客户端发来的数据//遍历客户端容器,移除无效客户端for (int i=0;i<socketList.count();i++){//判断客户端和服务器的连接状态//SocketState state() const;//客户端和服务器未连接的枚举值是0if(socketList.at(i)->state() == 0){//移除无效客户端socketList.removeAt(i);}}//遍历所有在线的客户端,找出待读数据的客户端for (int i = 0; i<socketList.count(); i++){//判断哪个客户端有数据待读if(socketList.at(i)->bytesAvailable() !=0){//读取数据QByteArray msg = socketList.at(i)->readAll();//将读取到的数据放入ui界面上ui->listWidget->addItem(QString::fromLocal8Bit(msg));//将数据广播给所有客户端for (int j=0;j<socketList.count();j++){socketList.at(j)->write(msg);}}}
}//启动服务器按钮对应的槽函数
void Widget::on_start_btn_clicked()
{//获取ui界面上的端口号quint16 port = ui->portEdit->text().toUInt();//将字符串转换为整型//服务器设置监听//bool listen(const QHostAddress &address = QHostAddress::Any,quint16 port=0);//参数1:监听的主机//参数2:监听的端口号//返回值:监听成功返回true  否则falseif(server->listen(QHostAddress::Any,port)){//监听成功QMessageBox::information(this,"","启动服务器成功!");}else{//监听失败QMessageBox::information(this,"","启动服务器失败!");return;}//此时如果有客户端发来连接请求,那么服务器就会自动发射一个newConnection()信号//将该信号连接到自定义的槽函数中,处理逻辑代码connect(server,&QTcpServer::newConnection,this,&Widget::newConnection_slot);}

客户端

头文件

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QTcpSocket>
#include <QMessageBox>
#include <QVBoxLayout>QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACEclass Widget : public QWidget
{Q_OBJECTpublic:Widget(QWidget *parent = nullptr);~Widget();private slots:void on_connectbtn_clicked();void on_sendbtn_clicked();void on_breakbtn_clicked();public slots:void connected_slot();void readyRead_slot();void disconnected_slot();private:Ui::Widget *ui;QTcpSocket *socket;//QString userName;
};
#endif // WIDGET_H

源文件

#include "widget.h"
#include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget), socket(new QTcpSocket(this))
{ui->setupUi(this);//初始化界面ui->msgEdit->setEnabled(false);ui->sendbtn->setEnabled(false);ui->breakbtn->setEnabled(false);//如果成功连接到服务器,那么客户端就会自动发射一个connected()信号//我们就可以将该信号连接到自定义的槽函数中处理逻辑代码,由于只需要连接一次,所以连接函数写在构造函数中connect(socket, &QTcpSocket::connected, this, &Widget::connected_slot);//程序运行至此,说明客户端成功与服务建立连接,如果服务器发来数据,那么客户端就会自动发射一个readyRead()信号//我们就可以将该信号连接到自定义的槽函数,读取数据。由于只需连接一次,所以连接函数写在构造函数中connect(socket, &QTcpSocket::readyRead, this, &Widget::readyRead_slot);//如果成功断开与服务器的连接,那么客户端就会自动发射一个disconnected的信号//我们就可以将该信号连接到自定义的槽函数,。由于只需连接一次,所以连接函数写在构造函数中connect(socket,&QTcpSocket::disconnected,this,&Widget::disconnected_slot);
}Widget::~Widget()
{delete ui;
}//readyRead()信号对应的槽函数
void Widget::readyRead_slot()
{static int row=0;//读取服务器发来的数据QByteArray msg = socket->readAll();//拆分出用户名QStringList list = (QString::fromLocal8Bit(msg)).split(":");//将数据放入ui界面上ui->listWidget->addItem(QString::fromLocal8Bit(msg));//根据用户名设置显示位置if(list[0]==userName){//是本人的消息    靠右显示ui->listWidget->item(row)->setTextAlignment(Qt::AlignRight);}else{//不是本人的消息    靠左显示ui->listWidget->item(row)->setTextAlignment(Qt::AlignLeft);}if(list[1]=="离开聊天室"||list[1]=="进入聊天室"){//居中显示ui->listWidget->item(row)->setTextAlignment(Qt::AlignCenter);}row++;
}//connected()信号对应的槽函数
void Widget::connected_slot()
{userName = ui->userNameEdit->text();QString msg = userName + ":进入聊天室";//将信息发送服务器socket->write(msg.toLocal8Bit());//连接服务器成功QMessageBox::information(this,"","连接成功!");//组件可用的相关设置ui->msgEdit->setEnabled(true);ui->sendbtn->setEnabled(true);ui->breakbtn->setEnabled(true);ui->userNameEdit->setEnabled(false);ui->ipEdit->setEnabled(false);ui->portEdit->setEnabled(false);ui->connectbtn->setEnabled(false);//程序运行至此,说明客户端成功与服务建立连接,如果服务器发来数据,那么客户端就会自动发射一个readyRead()信号//我们就可以将该信号连接到自定义的槽函数,读取数据。由于只需连接一次,所以连接函数写在构造函数中
}//disconnected()信号对应的槽函数
void Widget::disconnected_slot()
{//组件可用的相关设置ui->msgEdit->setEnabled(false);ui->sendbtn->setEnabled(false);ui->breakbtn->setEnabled(false);ui->userNameEdit->setEnabled(true);ui->ipEdit->setEnabled(true);ui->portEdit->setEnabled(true);ui->connectbtn->setEnabled(true);}//发送按钮对应的槽函数
void Widget::on_sendbtn_clicked()
{//获取ui界面的信息QString msg = ui->msgEdit->text();msg = userName + ":" + msg;//发送给服务器socket->write(msg.toLocal8Bit());//清空行编辑器ui->msgEdit->clear();
}//连接服务器按钮对应的槽函数
void Widget::on_connectbtn_clicked()
{//获取ui界面上的IP和端口号QString ip = ui->ipEdit->text();quint16 port = ui->portEdit->text().toUInt();//让客户端连接服务器//virtual void connectToHost(const QHostAddress &address, quint16 port, OpenMode mode = ReadWrite);//参数1:服务器的ip地址//参数2:端口号socket->connectToHost(ip,port);//如果成功连接到服务器,那么客户端就会自动发射一个connected()信号//将该信号连接到自定义的槽函数,书写逻辑代码。由于只需要连接一次,所以连接函数写在构造函数中}//断开连接按钮对应的槽函数
void Widget::on_breakbtn_clicked()
{QString msg = userName + ":离开聊天室";socket->write(msg.toLocal8Bit());//断开与服务器的连接//virtual void disconnectFromHost();socket->disconnectFromHost();//如果成功断开与服务器的连接,那么客户端就会自动发射一个disconnected的信号//我们就可以将该信号连接到自定义的槽函数,。由于只需连接一次,所以连接函数写在构造函数中
}

这篇关于8.21 QT的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1097036

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点

使用Qt编程QtNetwork无法使用

使用 VS 构建 Qt 项目时 QtNetwork 无法使用的问题 - 摘叶飞镖 - 博客园 (cnblogs.com) 另外,强烈建议在使用QNetworkAccessManager之前看看这篇文章: Qt 之 QNetworkAccessManager踏坑记录-CSDN博客 C++ Qt开发:QNetworkAccessManager网络接口组件 阅读目录 1.1 通用API函数

Qt多语种开发教程

Qt作为跨平台的开发工具,早已应用到各行各业的软件开发中。 今天讲讲,Qt开发的正序怎么做多语言开发。就是说,你设置中文,就中文显示;设置英语就英文显示,设置繁体就繁体显示,设置发育就显示法语等。 开发环境(其实多语种这块根环境没太大关系):win10,Qt.5.12.10 一.先用QtCreator创建一个简单的桌面程序 1.工程就随便命名“LanguageTest”,其他默认。 2.在设计师

Qt中window frame的影响

window frame 在创建图形化界面的时候,会创建窗口主体,上面会多出一条,周围多次一圈细边,这就叫window frame窗口框架,这是操作系统自带的。 这个对geometry的一些属性有一定影响,主要体现在Qt坐标系体系: 窗口当中包含一个按钮,这个按钮的坐标系是以父元素为参考,那么这个参考是widget本体作为参考,还是window frame作为参考,这两种参考体系都存在

【Qt】定时器事件

定时器事件 在之前学习QTimer中实现了定时器的功能,而在QTimer背后是QTimerEvent定时器事件进行支撑的。在QObject中提供了一个timeEvent这个函数。 startTimer启动定时器killTimer关闭定时器 Qt 中在进⾏窗⼝程序的处理过程中,经常要周期性的执⾏某些操作,或者制作⼀些动画效果,使⽤定 时器就可以实现。所谓定时器就是在间隔⼀定时间后,去执⾏某⼀

QT 编译报错:C3861: ‘tr‘ identifier not found

问题: QT 编译报错:C3861: ‘tr’ identifier not found 原因 使用tr的地方所在的类没有继承自 QObject 类 或者在不在某一类中, 解决方案 就直接用类名引用 :QObject::tr( )

在 Qt Creator 中,输入 /** 并按下Enter可以自动生成 Doxygen 风格的注释

在 Qt Creator 中,当你输入 /** 时,确实会自动补全标准的 Doxygen 风格注释。这是因为 Qt Creator 支持 Doxygen 以及类似的文档注释风格,并且提供了代码自动补全功能。 以下是如何在 Qt Creator 中使用和显示这些注释标记的步骤: 1. 自动补全 Doxygen 风格注释 在 Qt Creator 中,你可以这样操作: 在你的代码中,将光标放在

Qt: 详细理解delete与deleteLater (避免访问悬空指针导致程序异常终止)

前言 珍爱生命,远离悬空指针。 正文 delete 立即删除:调用 delete 后,对象会立即被销毁,其内存会立即被释放。调用顺序:对象的析构函数会被立即调用,销毁该对象及其子对象。无事件处理:如果在对象销毁过程中还涉及到信号和槽、事件处理等,直接 delete 可能会导致问题,尤其是在对象正在处理事件时。适用场景:适用于在确定对象已经不再被使用的情况下,并且不涉及异步处理或事件循环中的