在Qt平台上的网络应用编程原理

前言

        在网络通信方面的应用编程需要使用套接字(Socket),如在构建网站的服务器、游戏的服务器时。Qt提供了跨平台的类库QTcpServer、QTcpSocket及QUdpSocket供程序员使用,具体用途如下。

  • QTcpServer用于传输控制协议/网际协议(Transmission Control Protocol/internetProtocol,TCP/IP)通信,作为服务器端套接字使用。
  • QTcpSocket用于TCP/IP 通信,作为客户端套接字使用。
  • QUdpSocket用于用户数据报协议(User Datagram Protocol,UDP)通信,服务器端、客户端均使用此套接字。

        网络编程模块是 Qt的基本模块之一,在编程时需引入,具体方法是在.pro 文件中通过如下方式添加。

Qt+= network

 TCP/IP 原理

        Qt的 TCP Socket通信有服务器端、客户端之分。服务器端通过监听某个端口来确定是否有客户端连接到来,如果有连接到来,则建立新的 Socket连接;而客户端通过IP 连接服务器端当成功建立连接之后,就可进行数据传输了。TCP是一种面向连接的、可靠的、基于字节流的传输层协议。所谓面向连接就是在真正传输数据之前,需要与对方先建立连接,如果建立失败则不会传输数据。

        TCP 连接建立的过程被称为3次握手,过程如下:

  1. 客户端发送 SYN seq=x 报文给服务器端,客户端进入 SYN_SENT 状态。
  2. 服务端收到 SYN 报文后,发送 SYN seg=y,ACK=x+1 报文给客户端,服务器端进入 SYNRCVD 状态。
  3. 客户端收到服务器端的 SYN 报文后发送一个 ACK=y+1报文给服务器端,进入 ESTABLISHED状态。

         3 次握手完成,TCP 客户端与服务器端建立连接,正式开始传输数据。而在断开连接时需要4次挥手,具体过程如下(以客户端向服务器端断开连接为例)。

  1. 客户端调用 close() 向服务器端发送 FIN seg=x+2 ACK=y+1,表示数据发送完成。
  2. 服务器端接收 FIN seq=x+2 ACK=y+1,执行被动关闭。
  3. 一段时间后服务器端调用 close()向客户端发送 FIN seq=y+1。
  4. 客户端确认 FIN seg=y+1,断开连接成功。

        TCP 具有高可靠性,可保证传输数据的正确以及顺序,使用场景广泛,如超文本传输协议(HyperText Transfer Protocol,HTTP)就使用了 TCP。而Qt对TCP也提供了支持,以方便程序员设计网络相关的应用。


TCP Socket编程

        在Qt中是通过 Socket实现网络通信的,Socket可理解为程序在网络中的唯一标识。Qt实现的 TCP/IP 服务分为两部分,分别是服务器端和客户端

服务器端

1.通过QTcpServer()函数创建用于监听的 Socket:

m_server =new QTcpServer(this);

2. 将 Socket 设置为监听模式。

m_server->listen(QHostAddress::Any,8000);
// 表示监听本机网络的 80 号端口,如 127.0.0.1:80,Any 表示 IP 地址

3. 如果有连接到来,监听的 Socket会发出信号 newConnection。

connect(m_server, &QTcpServer::newConnection, this, &MainWindow::newConnection);

 4.接收连接,通过 nextPendingConnection()函数,返回一个 QTcpSocket 类型的 Socket对象(用于通信)。

m_client=m_server->nextPendingConnection();

 5.使用用于通信的 Socket 对象通信。

//连接信号,接收客户端数据
//&QTcpSocket::readyRead 是接收数据后发出的信号
//&MainWindow::readyRead 是自定义槽函数
connect(m_client, &QTcpSocket::readyRead, this, &MainWindow::readyRead);
  • 发送数据:write()
m_client->write("服务器连接成功 !!!");//回复客户端
  •  接收数据readAll()/read()。
QByteArray array=m_client->readAll();// 接收数据

客户端 

1.创建用于通信的 Socket。

server=new QTcpSocket(this);// 创建客户端 Socket

2. 连接服务器:connectToHost。

server->connectToHost(QHostAddress("127.0.0.1"),8000);//连接服务器

3. 连接成功后与服务器通信。

  • 发送数据:write()。
// 接收服务器端数据,slotReadyRead 是自定义槽函数
connect(server, &QTcpSocket::readyRead,this, &MainWindow::slotReadyRead);
  • 接收数据:readAll()/read()。
//发送数据,slotSendMsg 是自定义槽函数
connect(ui->pushButtorn, &QPushButton::clicked,this, &MainWindow::slotSendMsg);

 示例:

//服务器端
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QTcpServer>
#include <QTcpSocket>
#include <QByteArray>
MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);m_server=new QTcpServer(this);m_server->listen(QHostAddress::Any,8000);connect(m_server,&QTcpServer::newConnection,this,&MainWindow::newConnection);
}void MainWindow::newConnection(){if(m_client==NULL){m_client=m_server->nextPendingConnection();m_client->write("连接成功");connect(m_client,&QTcpSocket::readyRead,this,&MainWindow::readyRead);}
}void MainWindow::readyRead(){QByteArray array = m_client->readAll();ui->showtext->setText(QString(array));
}MainWindow::~MainWindow()
{delete ui;
}
//客户端
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QHostAddress>
#include<QMessageBox>
MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);server=new QTcpSocket(this);server->connectToHost(QHostAddress("127.0.0.1"),8000);connect(server,&QTcpSocket::readyRead,this,&MainWindow::slotReadyRead);connect(ui->on_btn_server,&QPushButton::clicked,this,&MainWindow::slotSendMsg);
}void MainWindow::slotReadyRead(){QByteArray array=server->readAll();QMessageBox::information(this,"Server Message",array);
}void MainWindow::slotSendMsg(){QString text=ui->textEdit->toPlainText();server->write(text.toUtf8());ui->textEdit->clear();
}MainWindow::~MainWindow()
{delete ui;
}

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

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

相关文章

竞赛 基于CNN实现谣言检测 - python 深度学习 机器学习

文章目录 1 前言1.1 背景 2 数据集3 实现过程4 CNN网络实现5 模型训练部分6 模型评估7 预测结果8 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于CNN实现谣言检测 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学长非常推荐&am…

欧姆龙61F系列液位开关使用教程(补水和排水)

欧姆龙61F系列液位开关使用教程(补水和排水) 本文以61F-LS-CP11-NRA型号的液位开关为例进行说明: 具体的选型文档可参考以下链接中的内容: OMRON欧姆龙-无浮标开关(紧凑插入型)61F-LS液位开关-选型样本说明 补水功能(供水) 如下图所示, 电机电源为3相AC220V; 控制电…

SSRF+Redis未授权getshell

SSRFRedis未授权getshell 1.前言 当一个网站具有ssrf漏洞&#xff0c;如果没有一些过滤措施&#xff0c;比如没过滤file协议&#xff0c;gophere协议&#xff0c;dict等协议&#xff0c;就会导致无法访问的内网服务器信息泄露&#xff0c;甚至可以让攻击者拿下内网服务器权限 …

Git分布式版本控制系统——Git常用命令(二)

五、Git常用命令————分支操作 同一个仓库可以有多个分支&#xff0c;各个分支相互独立&#xff0c;互不干扰 分支的相关命令&#xff0c;具体如下&#xff1a; git branch 查看分支 git branch [name] 创建分支&#x…

5. Mysql的binlog介绍

参考&#xff1a;InnoDB学习&#xff08;三&#xff09;之BinLog 1. BinLog介绍 BinLog又称为二进制日志&#xff0c;是MySQL服务层的数据日志&#xff0c;MySQL所有的存储引擎都支持BinLog。 BinLog记录了MySQL中的数据更新和可能导致数据更新的事件&#xff0c;可以用于主从…

轻量带屏解决方案之恒玄芯片移植案例

本文章基于恒玄科技BES2600W芯片的欧智通 Multi-modal V200Z-R开发板 &#xff0c;进行轻量带屏开发板的标准移植&#xff0c;开发了智能开关面板样例&#xff0c;同时实现了ace_engine_lite、arkui_ui_lite、aafwk_lite、appexecfwk_lite、HDF等部件基于OpenHarmony LiteOS-M内…

Flume配置案例@Source:Kafka,Channel:File,Sink:HDFS

创建flume配置文件 [atguiguhadoop104 flume]$ vim job/kafka_to_hdfs_log.conf 配置内容如下&#xff1a; --------------------- #定义组件 a1.sourcesr1 a1.channelsc1 a1.sinksk1 #配置source1 a1.sources.r1.type org.apache.flume.source.kafka.KafkaSource a1…

【联机不卡顿】幻兽帕鲁教你如何低成本0延迟畅玩 云服务器性价比选择方案 16G低至26/月

更新日期&#xff1a;4月14日&#xff08;腾讯云16G价格回调了&#xff01;京东云采购季持续进行&#xff09; 本文纯原创&#xff0c;侵权必究 《最新对比表》已更新在文章头部—腾讯云文档&#xff0c;文章具有时效性&#xff0c;请以腾讯文档为准&#xff01; 【腾讯文档实…

算法:位运算

算法&#xff1a;位运算 常见位运算操作基本题型模拟加法数字查找总结 常见位运算操作 在C/C中&#xff0c;有比较丰富的位运算操作符&#xff0c;常见的有&#xff1a; &&#xff1a;按位与 |&#xff1a;按位或 ~&#xff1a;按位取反 ^&#xff1a;按位异或 <<&a…

MySQ数据库: MySQL数据库的安装配置 ,图文步骤详细,一篇即可完成安装完成! MySQL数据库如何与客户端连接

LiuJinTao&#xff1a; 2024年4月14日 文章目录 MySQL的安装配置1. 下载2. 安装 三、 MySQL 启动与停止1. 第一种 方式&#xff1a;2. 第二种方式&#xff1a; 四、MySQL 客户端连接2. 方式二&#xff1a; MySQL的安装配置 1. 下载 官方下载网址&#xff1a;https://www.mysq…

代码随想录刷题随记21-回溯1

代码随想录刷题随记21-回溯1 回溯法解决的问题 回溯法&#xff0c;一般可以解决如下几种问题&#xff1a; 组合问题&#xff1a;N个数里面按一定规则找出k个数的集合 切割问题&#xff1a;一个字符串按一定规则有几种切割方式 子集问题&#xff1a;一个N个数的集合里有多少符…

可视化大屏C位图:​地理信息—地球焦点图

Hello&#xff0c;我是大千UI工场&#xff0c;本期可视化大屏的焦点图&#xff08;C位&#xff09;分享将地球作为焦点图的情形&#xff0c;欢迎友友们关注、评论&#xff0c;如果有订单可私信。 将地球作为可视化大屏焦点图可以有以下几个作用&#xff1a; 全球数据展示&…

SQL server 非聚集索引

CREATE NONCLUSTERED INDEX idx_dwd_kuoutlist2_dt_kuout_kg ON [dwd_kuoutlist2] (dt, kuout_kg,id) 这条SQL语句是在SQL Server中创建一个非聚集索引。具体含义如下&#xff1a; 1. **CREATE NONCLUSTERED INDEX**&#xff1a; - 这部分表明正在创建一个非聚集索引。非聚…

蓝桥杯嵌入式(G431)备赛笔记——DMA+ADC(单通道+多通道)

单通道&#xff1a; 开启循环模式&#xff0c;两个参数设为word u32 adc_tick0; u32 r37_value0; u32 r38_value0; float r37_volt0; float r38_volt0;//DMAADCvoid DMA_ADC() {if(uwTick-adc_tick<100) return;adc_tick uwTick;HAL_ADC_Start_DMA(&hadc2, &r37_v…

Python学习笔记19 - 类与对象

类的创建 对象的创建 类属性、类方法、静态方法 动态绑定属性和方法 面向对象的三大特征 封装&#xff1a;提高程序的安全性 继承&#xff1a;提高代码的复用性 多态&#xff1a;提高程序的可扩展性和可维护性 类的常用的特殊属性 类的常用的特殊方法 –add–() –len–() –…

Java中创建多线程的方法

继承Thread类&#xff0c;对该类进行new一个实例&#xff0c;对实例调用start方法&#xff0c;重写run方法。 缺点&#xff1a;单继承&#xff0c;无法继承 public class myThread extends Thread {public static void main(String[] args) {myThread myThread new myThread()…

Netty学习——实战篇1 BIO、NIO入门demo 备注

1 BIO 实战代码 Slf4j public class BIOServer {public static void main(String[] args) throws IOException {//1 创建线程池ExecutorService threadPool Executors.newCachedThreadPool();//2 创建ServerSocketServerSocket serverSocket new ServerSocket(8000);log.in…

【嵌入式基础知识学习】AD/DA—数模/模数转换

AD/DA—数模/模数转换概念 数字电路只能处理二进制数字信号&#xff0c;而声音、温度、速度和光线等都是模拟量&#xff0c;利用相应的传感器&#xff08;如声音用话筒&#xff09;可以将它们转换成模拟信号&#xff0c;然后由A/D转换器将它们转换成二进制数字信号&#xff0c…

6个Python迭代器及生成器酷炫实例

大家好&#xff0c;今天我们要一起探索Python中的六个神奇工具&#xff1a;迭代器和生成器。它们就像魔法棒&#xff0c;能让我们优雅地处理大量数据&#xff0c;既节省内存又高效。别担心&#xff0c;我会用最接地气的语言和简单易懂的代码示例来带你一探究竟&#xff01; 1.…

音视频学习—音视频理论基础(1)

音视频学习—音视频理论基础&#xff08;1&#xff09; 一、音视频处理流程1.1 音频处理流程1.2 视频处理流程1.3 音视频数据流转1.4 为什么音视频采集完之后&#xff0c;不能直接传输&#xff0c;要进行编码&#xff1f;1.5 模数转换1.6 PCM1.7 WAV 总结 一、音视频处理流程 音…