当前位置: 首页 > news >正文

[Qt][Qt 网络][下]详细讲解

目录

  • 1.TCP Socket
    • 1.核心API概览
    • 2.回显服务器
    • 3.回显客户端
  • 2.HTTP Client
  • 3.其他模块


1.TCP Socket

1.核心API概览

  • 核⼼类是两个:QTcpServerQTcpSocket
  • QTcpServer用于监听端口,和获取客户端连接
    • listen(const QHostAddress&, quint16 port):绑定指定的地址和端⼝号,并开始监听
      • 成员方法
      • 对标原⽣API:bind() 和listen()
    • nextPendingConnection():从系统中获取到⼀个已经建⽴好的tcp连接
      • 返回⼀个QTcpSocket,表⽰这个客⼾端的连接
        • 通过这个socket对象完成和客⼾端之间的通信
      • 成员方法
      • 对标原⽣API:accept()
    • newConnection():有新的客⼾端建⽴连接好之后触发
      • 信号
      • 类似于IO多路复用的通知机制
  • QTcpSocket⽤于客⼾端和服务器之间的数据交互
    • readAll():读取当前接收缓冲区中的所有数据,返回QByteArray对象
      • 成员方法
      • 对标原生API:read()
    • write(const QByteArray &):把数据写⼊socket中
      • 成员方法
      • 对标原生API:write()
    • deleteLater():暂时把socket对象标记为⽆效,Qt会在下个事件循环中析构释放该对象
      • 成员方法
      • 类似于"半⾃动化的垃圾回收"
    • readyRead():有数据到达并准备就绪时触发
      • 信号
      • 类似于IO多路复⽤中的通知机制
    • disconnected():连接断开时触发
      • 信号
      • 类似于IO多路复⽤中的通知机制

2.回显服务器

// 构造函数中
{server = new QTcpServer(this);// 通过信号槽, 指定如何处理连接.connect(server, &QTcpServer::newConnection, this, &Widget::ProcessConnection);// 绑定并监听端口号if (!ret = server->listen(QHostAddress::Any, 2333)) {QMessageBox::critical(this, "服务器启动失败!", server->errorString());exit(1);}
}void Widget::ProcessConnection()
{// 通过 server 拿到一个 socket 对象, 通过这个对象来和客户端进行通信QTcpSocket* clientSocket = tcpServer->nextPendingConnection();QString log = "[" + clientSocket->peerAddress().toString() + ":" + QString::number(clientSocket->peerPort()) + "] 客户端上线!";ui->listWidget->addItem(log);// 通过信号槽, 来处理客户端发来请求的情况connect(clientSocket, &QTcpSocket::readyRead, this, [=]() {// a) 读取出请求数据,此处 readAll 返回的是 QByteArray, 通过赋值转成 QStringQString request = clientSocket->readAll();// b) 根据请求处理响应const QString& response = Process(request);// c) 把响应写回到客户端clientSocket->write(response.toUtf8());// d) 把上述信息记录到日志中.QString log = "[" + clientSocket->peerAddress().toString() + ":" + QString::number(clientSocket->peerPort()) + "] "+ " req: " + request + ", resp: " + response;ui->listWidget->addItem(log);});// 通过信号槽, 来处理客户端断开连接的情况.connect(clientSocket, &QTcpSocket::disconnected, this, [=]() {// a) 把断开连接的信息通过日志显示出来.QString log = "[" + clientSocket->peerAddress().toString() + ":" + QString::number(clientSocket->peerPort()) + "] 客户端下线!";ui->listWidget->addItem(log);// b) 手动释放 clientSocket. 直接使用 delete 是下策, 使用 deleteLater 更加合适clientSocket->deleteLater();});
}QString Widget::Process(const QString request)
{return request;
}

3.回显客户端

// 构造函数中
{socket = new QTcpSocket(this);// 服务器建立连接,此处是非阻塞的IO通信socket->connectToHost("127.0.0.1", 2333);// 连接信号槽, 处理响应connect(socket, &QTcpSocket::readyRead, this, [=]() {// a) 读取出响应内容QString response = socket->readAll();// b) 把响应内容显示到界面上.ui->listWidget->addItem("服务器说: " + response);});// 等待连接建立的结果. 确认是否连接成功.if (!socket->waitForConnected()) {QMessageBox::critical(this, "连接服务器出错", socket->errorString());exit(1);}
}void Widget::on_pushButton_clicked()
{const QString& text = ui->lineEdit->text();socket->write(text.toUtf8());ui->listWidget->addItem("客户端说: " + text);ui->lineEdit->setText("");
}

2.HTTP Client

  • Qt只提供了Http客户端,并没有提供Http服务端的库
  • 关键类主要是三个:QNetworkAccessManager , QNetworkRequest , QNetworkReply
  • QNetworkAccessManager提供了HTTP的核⼼操作
    • get(const QNetworkRequest&):发起⼀个HTTP GET请求,返回QNetworkReply对象
      • 本身不是阻塞函数,只负责发出去请求,不负责等请求回来
      • 收是QNetworkReply负责的,会收到finished信号
    • post(const QNetworkRequest&, const QByteArray&):发起⼀个HTTP POST请求,返回QNetworkReply对象
  • QNetworkRequest表⽰⼀个HTTP请求(不含body),如果需要发送⼀个带有body的请求(⽐如post),会在QNetworkAccessManagerpost()中通过单独的参数来传⼊body
    • QNetworkRequest(const QUrl&):通过URL构造⼀个HTTP请求
    • setHeader(QNetworkRequest::KnownHeaders header, const QVariant& value):设置请求头
      • QNetworkRequest::KnownHeaders是⼀个枚举类型,常⽤取值
        • ContentTypeHeader:描述body的类型
        • ContentLengthHeader:描述body的⻓度
        • LocationHeader:⽤于重定向报⽂中指定重定向地址
        • CookieHeader:设置cookie
        • UserAgentHeader:设置User-Agent
  • QNetworkReply表⽰⼀个HTTP响应,这个类同时也是QIODevice的⼦类
    • error():获取出错状态
    • errorString():获取出错原因的⽂本
    • readAll():读取响应body
    • header(QNetworkRequest::KnownHeaders header):读取响应指定header的值
    • finished():为一个信号,在客⼾端收到完整的响应数据之后触发
  • 示例
    // 构造函数中
    {manager = new QNetworkAccessManager(this);
    }void Widget::on_pushButton_clicked()
    {// 1. 获取到输入框中的 urlQUrl url(ui->lineEdit->text());// 2. 构造一个 HTTP 请求对象QNetworkRequest request(url);// 3. 发送请求QNetworkReply* response = manager->get(request);// 4. 通过信号槽, 来处理响应connect(response, &QNetworkReply::finished, this, [=]() {if (response->error() == QNetworkReply::NoError) {QString html = response->readAll();ui->plainTextEdit->setPlainText(html);} else {ui->plainTextEdit->setPlainText(response->errorString());}// 需要对 response 进行手动释放.response->deleteLater();});
    }
    

3.其他模块

  • Qt中还提供了FTP,DNS,SSL等⽹络相关的组件⼯具,有需要可以具体学习,查阅官方文档
http://www.lryc.cn/news/426820.html

相关文章:

  • 十三、OpenCVSharp的目标检测
  • STM32标准库学习笔记-6.定时器-输入捕获
  • vue前端可以完整的显示编辑子级部门,用户管理可以为用户分配角色和部门?
  • 量化交易的基石:ExchangeSdk
  • 【区块链+金融服务】基于区块链的一站式绿色金融开放平台 | FISCO BCOS应用案例
  • 使用Python实现深度学习模型:智能娱乐与虚拟现实技术
  • 亚马逊云科技产 Amazon Neptune 图数据库服务体验
  • 【网络安全】重置密码token泄露,实现账户接管
  • 计算机基础知识复习8.13
  • Unity URP无光照下Shadow 制作 <二> 合批处理
  • 微乐校园pf
  • 文件其他相关函数
  • SQLALchemy ORM 的关联关系之 ORM 中的多对多
  • sdkman install慢,采用squid代理
  • 实时监控Windows服务器:使用Prometheus和Grafana的终极方案
  • 【文科生能看懂的】牛顿二项式定理
  • Fly Catcher:通过监测恶意信号来检测飞机欺骗
  • 计算机网络——HTTP协议详解(上)
  • 十九、中介者模式
  • 编程参考 - 头文件中使用static inline
  • Uniapp使用antd组件库
  • 计算机毕业设计选题推荐-高校实验室管理系统-Java/Python项目实战
  • nest定义响应码message文本
  • Java | Leetcode Java题解之第342题4的幂
  • 【日常开发】java中一个list对象集合 将字段a为 大豆 小麦 玉米等元素放在最前面 并组成新集合
  • C++ 设计模式——原型模式
  • 【Harmony OS 4.0】待办列表案例
  • 快速把文件名统计到excel表的方法
  • 开源通用验证码识别OCR —— DdddOcr 源码赏析(一)
  • 上升ECMAScript性能优化技巧与陷阱(下)