Обмен технологиями

Сетевое программирование Qt http

2024-07-12

한어Русский языкEnglishFrançaisIndonesianSanskrit日本語DeutschPortuguêsΕλληνικάespañolItalianoSuomalainenLatina

Цели обучения: сетевое программирование Qt HTTP.

Учебный контент

1. Http — это аббревиатура протокола передачи гипертекста, который определяет характеристики связи между браузерами и веб-серверами. Это простой протокол запроса-ответа, который обычно работает поверх TCP.

Функция: он определяет спецификации передачи информации между WWW-сервером и браузером и представляет собой соглашение, которое соблюдают обе стороны.
2.Как работает http: HTTP основан на модели клиент/сервер и ориентирован на соединение. Процесс обработки HTTP-транзакций

  1. Клиент устанавливает соединение с сервером;
  2. Клиент отправляет запрос серверу;
  3. Сервер принимает запрос и возвращает соответствующий файл в качестве ответа по запросу:
  4. Клиент и сервер закрывают соединение.

Qt реализует операции HTTP двумя способами.

1. Используйте классы QNetworkAccessManager и QNetworkReply:

  1. - QNetworkAccessManager — это менеджер доступа к сети Qt, отвечающий за отправку и получение запросов HTTP/HTTPS.
  2. - Инициировать GET/POST и другие запросы через QNetworkAccessManager::get/post и т. д. и возвращать объекты QNetworkReply.
  3. - Получите код ответа, информацию заголовка, содержимое и т. д. от QNetworkReply. Реализуйте синхронные и асинхронные запросы.

2. Используйте классы QHttpEngine и QHttpNetworkRequest/Reply (нижний уровень):

  1. — QHttpEngine отвечает за управление HTTP-соединениями и обработку запросов.
  2. - Используйте QHttpNetworkRequest для создания запроса и отправки его через QHttpEngine.
  3. — Проанализируйте возвращенный QHttpNetworkReply, чтобы получить ответ.

Вообще говоря, для простых операций HTTP достаточно использования высокоуровневого интерфейса QNetworkAccessManager. Он удобно справляется с общими потребностями.

Если вам нужен контроль более низкого уровня, например настройка механизма HTTP или деталей запроса/ответа, вы можете использовать классы низкого уровня, такие как QHttpEngine.

Короче говоря, Qt предоставляет полную реализацию HTTP-клиента, и разработчики могут выбрать более подходящий интерфейс для операций сетевого взаимодействия в зависимости от реальных потребностей. Это обеспечивает хорошую поддержку разработки приложений Qt.

Подведем итог:

Первый тип: менеджер сетевых подключений QNetworkAccessManager, 2 действия: получить/отправить для получения или отправки запроса, а возвращаемый объект QNetworkReply содержит различную информацию о соединении.

Второй тип: QHttpEngine только управляет запросами на подключение и не несет ответственности за их отправку.

  • Используйте QHttpNetworkRequest для создания запроса и отправки его через QHttpEngine.

  • Проанализируйте возвращенный QHttpNetworkReply, чтобы получить ответ.

Эффекты проекта и основной код

HTTP-ридер

http-сервер

основной код

HTTP-ридер

  1. #ifndef MAINWINDOW_H
  2. #define MAINWINDOW_H
  3. #include <QMainWindow>
  4. #include <QtNetwork> // 提供编程TCP/IP客户端和服务器的类
  5. #include <QUrl> // 提供接口使用URLs
  6. QT_BEGIN_NAMESPACE
  7. namespace Ui { class MainWindow; }
  8. QT_END_NAMESPACE
  9. class MainWindow : public QMainWindow
  10. {
  11. Q_OBJECT
  12. public:
  13. MainWindow(QWidget *parent = nullptr);
  14. ~MainWindow();
  15. private slots:
  16. void on_pushButton_clicked();
  17. private:
  18. Ui::MainWindow *ui;
  19. QNetworkAccessManager* accesssMgr;
  20. };
  21. #endif // MAINWINDOW_H
  22. #include "mainwindow.h"
  23. #include "ui_mainwindow.h"
  24. MainWindow::MainWindow(QWidget *parent)
  25. : QMainWindow(parent)
  26. , ui(new Ui::MainWindow)
  27. {
  28. ui->setupUi(this);
  29. accesssMgr =new QNetworkAccessManager(this);
  30. QObject::connect(accesssMgr,QOverload<QNetworkReply*>::of(&QNetworkAccessManager::finished),this,
  31. [this](QNetworkReply* reply){
  32. QString read=reply->readAll();
  33. ui->textBrowser->setText(read);
  34. reply->deleteLater(); //释放
  35. });
  36. }
  37. MainWindow::~MainWindow()
  38. {
  39. delete ui;
  40. }
  41. void MainWindow::on_pushButton_clicked()
  42. {
  43. ui->label->setText("数据正在下载中,请耐心等待......");
  44. QString url=ui->textEdit->toPlainText();
  45. qDebug()<<url;
  46. accesssMgr->get(QNetworkRequest(QUrl(url)));
  47. }

http-сервер

  1. #include "httpserver.h"
  2. HttpServer::HttpServer(QObject *parent) : QObject(parent)
  3. {
  4. server =new QTcpServer(this);
  5. //连接回调
  6. connect(server,&QTcpServer::newConnection,this,&HttpServer::MyselfNewConnectCalllSlot);
  7. //开始监听
  8. if(!server->listen(QHostAddress::Any,8088))
  9. {
  10. qDebug()<<"n致命错误:Web服务器没有启动,请重新检查!"<<endl;
  11. }
  12. else
  13. {
  14. qDebug()<<"n正常启动:Web服务器端口:8088,等待客户端连接......"<<endl;;
  15. }
  16. }
  17. void HttpServer::MyselfNewConnectCalllSlot(){
  18. socket = server->nextPendingConnection(); //获取连接
  19. while(!(socket->waitForReadyRead(100))); //等待套接字是否有可读数据。 一直阻塞线程等待,直到有可读数据到达套接字,或者超时返回。
  20. QString webdata;
  21. webdata.resize(1000);
  22. webdata=socket->read(1000);
  23. qDebug()<<"正常运行:从浏览器读取数据信息......"<<webdata;
  24. //封装http协议
  25. socket->write("HTTP/1.1 200 OKrn");
  26. socket->write("Content-Type: text/htmlrn");
  27. socket->write("Connection: closern");
  28. socket->write("Refresh: 3rnrn"); // 每秒刷新Web浏览器
  29. socket->write("<!DOCTYPE>"
  30. "<html>"
  31. "<header>"
  32. "<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>"
  33. "<title>HttpServer</title>"
  34. "</header>"
  35. "<body>客户端已经连接HttpSever服务器秒数为:");
  36. QByteArray byte;
  37. static qint16 icount=0;
  38. byte.setNum(icount++);
  39. socket->write(byte);
  40. socket->write("</html>");
  41. /*
  42. flush() 不会等待数据真正发送出去,只是把数据从本地缓冲区 flush 到内核缓冲区。
  43. 如果输出缓冲区没有数据,flush() 有可能不做任何操作。
  44. 调用 flush() 可以确保目前为止写到 socket 的所有数据都发送给对端,而不是 buffered 在本地。
  45. 一般在发送完请求或应答后调用,保证数据及时发送出去。
  46. */
  47. socket->flush(); //刷新 socket 对象的输出缓冲区。 并发送
  48. connect(socket, &QTcpSocket::disconnected, socket, &QTcpSocket::deleteLater);
  49. socket->disconnectFromHost();
  50. }

 

Подведем итог

http — это протокол прикладного уровня, основанный на TCP, поэтому будут использоваться классы TCP, а инкапсуляция протокола фиксирована.Дизайн основной функции-члена

  1. 请求:
  2. GET /index.html HTTP/1.1
  3. Host: www.example.com
  4. User-Agent: Mozilla/5.0...
  5. Accept: text/html
  6. 响应:
  7. HTTP/1.1 200 OK
  8. Date: Mon, 27 Jan 2020 08:12:31 GMT
  9. Server: Apache/2.4.1 (Unix)
  10. Last-Modified: Wed, 22 Jul 2020 19:15:56 GMT
  11. ETag: "49092-5b6-52c65aa32f280"
  12. Accept-Ranges: bytes
  13. Content-Length: 438
  14. Connection: close
  15. Content-Type: text/html; charset=UTF-8
  16. <html>
  17. <head>
  18. <title>Welcome to example.com!</title>
  19. </head>
  20. <body>
  21. <h1>Hello World!</h1>
  22. <p>This is the hello world page for an example website.</p>
  23. </body>
  24. </html>
  25. 这个示例演示了一个简单的HTTP请求和响应过程:
  26. 客户端发送GET请求获取index.html页面
  27. 服务器返回200状态码,表示请求成功
  28. 返回各种报头如日期、服务器信息等
  29. 返回HTML页面内容作为响应体

 

Основными классами для программирования Qt HTTP являются QNetworkAccessManager, QNetworkRequest и QNetworkReply.

QNetworkAccessManager:

  • Управляйте подключениями и запросами к сети. Отправка и получение данных.

  • Предоставляет расширенные функции, такие как get(), post() и put(), для инициации HTTP-запросов.

  • Сигналы:finished(), authenticateRequired().

QNetworkRequest:

  • Представляет HTTP-запрос. Содержит такие атрибуты, как URL-адрес, заголовки и т. д.

  • Установите метод запроса, тип контента, информацию заголовка и т. д.

QNetworkОтвет:

  • Представляет ответ HTTP. Возвращаемый код состояния и содержимое.

  • Сигналы: downloadProgress(), error() и т.д. уведомляют о результатах обработки.

  • Функциональные функции, такие как readAll(), читают содержимое ответа.

Основные шаги:

  1. Создайте экземпляр объекта QNetworkAccessManager.

  2. Создайте объект QNetworkRequest, чтобы установить URL-адрес и свойства.

  3. Вызовите такие функции, как QNetworkAccessManager::get() или post(), чтобы сделать запрос.

  4. Прослушайте сигналы QNetworkReply, чтобы получить результаты.

  5. Прочитайте содержимое через QNetworkReply или обработайте ошибки.

  6. Удалите QNetworkReply и запросите объекты, чтобы предотвратить утечку памяти.

Наконец, прикрепите ссылку на исходный код
Если это было полезно для вас, пожалуйста, поставьте мне звезду

Демонстрация Qt: изучение процесса qt (gitee.com)