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

2024-04-14 17:52

本文主要是介绍在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;
}

这篇关于在Qt平台上的网络应用编程原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

QT Creator配置Kit的实现示例

《QTCreator配置Kit的实现示例》本文主要介绍了使用Qt5.12.12与VS2022时,因MSVC编译器版本不匹配及WindowsSDK缺失导致配置错误的问题解决,感兴趣的可以了解一下... 目录0、背景:qt5.12.12+vs2022一、症状:二、原因:(可以跳过,直奔后面的解决方法)三、解决方

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

Python中的filter() 函数的工作原理及应用技巧

《Python中的filter()函数的工作原理及应用技巧》Python的filter()函数用于筛选序列元素,返回迭代器,适合函数式编程,相比列表推导式,内存更优,尤其适用于大数据集,结合lamb... 目录前言一、基本概念基本语法二、使用方式1. 使用 lambda 函数2. 使用普通函数3. 使用 N

MyBatis-Plus 与 Spring Boot 集成原理实战示例

《MyBatis-Plus与SpringBoot集成原理实战示例》MyBatis-Plus通过自动配置与核心组件集成SpringBoot实现零配置,提供分页、逻辑删除等插件化功能,增强MyBa... 目录 一、MyBATis-Plus 简介 二、集成方式(Spring Boot)1. 引入依赖 三、核心机制