【QT HTTP】使用QtNetwork模块制作基于HTTP请求的C/S架构

目录

  • 0 引言
  • 1 HTTP基本知识
    • 1.1 请求类型
    • 1.2 HTTP请求报文格式
    • 1.3 HTTP响应报文格式
    • 1.4 拓展:GET vs POST 请求方法
      • GET请求
        • 请求报文:
        • 响应报文
      • POST请求
        • 请求报文
        • 响应报文
      • 其他注意事项
      • 示例:
        • GET请求示例
        • POST请求示例
  • 2 实战
    • 2.1 QtNetwork模块介绍
    • 2.2 编程实现HTTP客户端
    • 2.3 编程实现HTTP服务器

请添加图片描述

  • 🙋‍♂️ 作者:海码007
  • 📜 专栏:C++专栏
  • 💥 标题:【QT HTTP】使用QtNetwork模块制作基于HTTP请求的C/S架构
  • ❣️ 寄语:书到用时方恨少,事非经过不知难。
  • 🎈 最后:文章作者技术和水平有限,如果文中出现错误,希望大家能指正!

0 引言

  • 最近项目涉及到网络HTTP相关内容,需要处理客户端发送的POST、GET等请求。所以本文使用QT的QtNetwork模块来实现一个简单的发送POST、GET请求的客户端,然后响应POST、GET请求的服务器。
  • 本文涉及到一些HTTP的基本知识,可以参考博主之前的文章:计算机网络HTTP协议

1 HTTP基本知识

1.1 请求类型

HTTP(Hypertext Transfer Protocol)协议 定义了多种请求方法,也称为HTTP请求类型或HTTP动词。这些请求方法表示客户端希望对特定资源执行的操作。以下是常见的HTTP请求类型、其功能和应用场景:

  1. GET:

    • 功能: 用于从服务器获取指定资源的信息。请求的参数通常附在URL后面,通过查询字符串传递。
    • 应用场景: 用于查看网页、下载文件、获取数据等。是幂等的,不应该对服务器产生影响。(我们输入一个网址,其实就是从服务器获得一个HTML文件,然后浏览器内核再根据其将内容绘制出来)
  2. POST:

    • 功能: 用于向服务器提交数据,通常用于表单提交。请求的参数通常包含在请求体中。
    • 应用场景: 用于创建新资源、提交表单数据、上传文件等。可能对服务器产生影响。不是幂等的,多次相同的POST请求可能产生不同的结果。
  3. PUT:

    • 功能: 用于向服务器上传新资源,或者更新已存在的资源。请求的参数通常包含在请求体中。
    • 应用场景: 用于创建或更新资源。是幂等的,多次相同的PUT请求应该产生相同的结果。
  4. DELETE:

    • 功能: 用于请求服务器删除指定的资源
    • 应用场景: 用于删除指定资源。是幂等的,多次相同的DELETE请求应该产生相同的结果。
  5. PATCH:

    • 功能: 用于对资源进行部分更新。请求的参数通常包含在请求体中,表示对资源的局部修改。
    • 应用场景: 用于对资源进行局部更新,而不是替换整个资源。
  6. HEAD:

    • 功能: 类似于GET请求,但服务器只返回响应头,不返回实体主体。常用于检查资源的元信息,如是否存在、是否已经修改等。
    • 应用场景: 用于获取资源的头部信息,而不需要获取整个资源的内容。
  7. OPTIONS:

    • 功能: 用于获取目标资源支持的通信选项。客户端可以通过这个方法了解服务器支持的方法。
    • 应用场景: 用于确定服务器支持的方法,以及支持的头信息等。
  8. TRACE:

    • 功能: 用于在目标服务器上执行一个消息环回测试,客户端发送的请求会在最终的服务器上返回,用于诊断和调试。
    • 应用场景: 主要用于网络诊断,通常不会在实际应用中直接使用。

选择适当的HTTP请求类型取决于具体的操作和业务需求。每种请求类型都有其独特的功能和应用场景,使其适用于不同的情境。

1.2 HTTP请求报文格式

HTTP请求报文是客户端发送给服务器的文本信息,包含请求的各种参数和头信息。它的基本格式如下:

<Method> <Request-URI> <HTTP-Version>
<Headers>

<Optional Request Body>

其中,各部分的含义如下:

  • <Method>:HTTP请求方法,例如GET、POST、PUT等。
  • <Request-URI>:请求的资源标识符,通常是一个URL。
  • <HTTP-Version>:使用的HTTP协议版本,例如HTTP/1.1。
  • <Headers>:包含多行的头部信息,每行都包含一个头字段和对应的值。
  • <Optional Request Body>:可选的请求体,用于包含请求时需要发送的数据,例如POST请求中的表单数据。

以下是一个具体的例子:

GET /index.html HTTP/1.1
Host: www.example.com
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:99.0) Gecko/20100101 Firefox/99.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8
Accept-Language: en-US,en;q=0.5
Connection: keep-alive
Upgrade-Insecure-Requests: 1

在这个例子中:

  • 请求方法是GET。
  • 请求的资源标识符是/index.html
  • 使用的HTTP协议版本是HTTP/1.1。
  • 请求头部包含了HostUser-AgentAccept等字段,每个字段都以<header-name>: <header-value>的形式呈现。
  • 由于GET请求通常不包含请求体,因此没有<Optional Request Body>部分。

对于包含请求体的请求,例如POST请求,请求体会紧随请求头部,并用一个空行分隔。例如:

POST /submit-form HTTP/1.1
Host: www.example.com
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:99.0) Gecko/20100101 Firefox/99.0
Content-Type: application/x-www-form-urlencoded
Content-Length: 23

username=johndoe&password=123

在这个例子中,请求体包含了表单数据username=johndoe&password=123,并通过Content-Type头字段指定了数据的格式。

1.3 HTTP响应报文格式

HTTP响应报文是服务器返回给客户端的文本信息,包含了服务器对客户端请求的响应。其基本格式如下:

<HTTP-Version> <Status-Code> <Reason-Phrase>
<Headers>

<Optional Response Body>

其中,各部分的含义如下:

  • <HTTP-Version>:使用的HTTP协议版本,例如HTTP/1.1。
  • <Status-Code>:一个三位数的状态码,表示服务器对请求的处理结果。
  • <Reason-Phrase>:状态码的文本描述,描述了状态码的原因。
  • <Headers>:包含多行的头部信息,每行都包含一个头字段和对应的值。
  • <Optional Response Body>:可选的响应体,用于包含服务器返回给客户端的数据。

以下是一个具体的例子:

HTTP/1.1 200 OK
Date: Mon, 15 Nov 2023 12:00:00 GMT
Server: Apache
Content-Type: text/html
Content-Length: 1234
Connection: keep-alive

<html>
  <head>
    <title>Hello, World!</title>
  </head>
  <body>
    <h1>Welcome to my website!</h1>
  </body>
</html>

在这个例子中:

  • 使用的HTTP协议版本是HTTP/1.1。
  • 状态码是200,表示请求成功。
  • 原因短语是"OK",为状态码的文本描述。
  • 响应头部包含了DateServerContent-Type等字段。
  • 由于这是一个简单的HTML响应,响应体包含了一个HTML文档。

对于包含响应体的响应,例如HTML页面、JSON数据等,响应体会紧随响应头部,并用一个空行分隔。响应体的格式和内容取决于服务器的实际响应。

1.4 拓展:GET vs POST 请求方法

GET和POST请求在HTTP中的请求报文和响应报文中有一些区别,这主要涉及到数据的传递方式和一些特定的语义约定。


GET请求

请求报文:
  1. 参数传递: GET请求的参数通常附在URL的查询字符串中,通过?&符号进行连接,例如:http://example.com/resource?param1=value1&param2=value2
  2. 请求体: GET请求通常没有请求体,因为它用于请求资源,而不是向服务器提交数据。
响应报文
  1. 响应体: GET请求的响应体包含了服务器返回的资源数据。

POST请求

请求报文
  1. 参数传递: POST请求的参数通常包含在请求体中,而不是在URL中,特别是用于提交表单数据或上传文件等场景。
  2. 请求体: POST请求的请求体包含了客户端提交给服务器的数据。

响应报文
  1. 响应体: POST请求的响应体包含了服务器对提交的数据的处理结果。

其他注意事项

  1. 安全性: POST请求的数据包含在请求体中,相对于GET请求,POST请求具有更好的安全性,因为它不会在URL中明文传递敏感信息。
  2. 幂等性: GET请求是幂等的,多次相同的GET请求应该产生相同的结果。POST请求是非幂等的,多次相同的POST请求可能会产生不同的结果。

示例:

GET请求示例

请求报文:

GET /resource?param1=value1&param2=value2 HTTP/1.1
Host: example.com

响应报文:

HTTP/1.1 200 OK
Content-Type: text/html
Content-Length: 1234

<!DOCTYPE html>
<html>
  <head>
    <title>GET Response</title>
  </head>
  <body>
    <h1>This is the response to a GET request.</h1>
  </body>
</html>

POST请求示例

请求报文:

POST /submit-form HTTP/1.1
Host: example.com
Content-Type: application/x-www-form-urlencoded
Content-Length: 27

param1=value1&param2=value2

响应报文:

HTTP/1.1 200 OK
Content-Type: application/json
Content-Length: 45

{"status": "success", "message": "POST response"}

总的来说,GET和POST请求的区别主要在于参数传递的方式、请求体的内容和请求的语义。GET适用于获取资源,而POST适用于向服务器提交数据。

2 实战

2.1 QtNetwork模块介绍

QtNetwork模块是Qt中用于网络编程的模块,提供了一系列用于处理网络通信的类和工具。以下是QtNetwork模块的一些主要功能:

  1. TCP和UDP通信: 提供QTcpSocketQUdpSocket等类,用于实现TCP和UDP协议的通信。这些类使得在Qt应用程序中创建和管理网络连接变得相对简单。

  2. HTTP客户端和服务器: 提供QNetworkAccessManager类,用于实现HTTP协议的客户端功能。它支持GET、POST等HTTP请求方法,并允许异步地发送和接收HTTP请求。

  3. 网络请求和响应处理: 提供QNetworkRequestQNetworkReply等类,用于构建和处理网络请求。这些类提供了丰富的功能,包括请求头的设置、数据的传输和响应的处理等。

  4. FTP客户端: 提供QFtp类,用于实现FTP协议的客户端功能。它允许在Qt应用程序中进行文件传输操作。

  5. 网络代理: 支持网络代理设置,可以通过QNetworkProxy类配置网络代理,以便在需要时通过代理服务器进行网络通信。

  6. 网络协议支持: QtNetwork模块支持各种网络协议,包括IPv4和IPv6,SSL/TLS等。这使得Qt应用程序能够适应多种网络环境和安全需求。

  7. 网络状态监控: 提供QNetworkConfigurationQNetworkConfigurationManager类,用于监控和管理网络配置,以便在应用程序中适应不同的网络状态。

  8. 网络缓存: 提供QNetworkDiskCache等类,用于实现网络缓存,以提高应用程序的性能并减少对网络资源的依赖。

这些功能使QtNetwork成为一个强大的网络编程工具,适用于开发涉及网络通信的各种应用,从简单的客户端到复杂的服务器应用。

2.2 编程实现HTTP客户端

根据上述描述,可以知道,使用 QTcpSocketQUdpSocketQNetworkAccessManagerQNetworkRequestQNetworkReply等类可以实现简单的HTTP客户端。

接下来是代码:

#include <QCoreApplication>
#include <QNetworkAccessManager>
#include <QNetworkRequest>
#include <QNetworkReply>
#include <QDebug>
#include <QUrlQuery>

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

    // 创建网络访问管理器
    QNetworkAccessManager manager;

    // 创建HTTP请求
    QNetworkRequest getRequest(QUrl("http://example.com"));

    // 发送GET请求
    QNetworkReply *getReply = manager.get(getRequest);

    // 处理GET请求完成的信号
    QObject::connect(getReply, &QNetworkReply::finished, [&]() {
        if (getReply->error() == QNetworkReply::NoError) {
            qDebug() << "GET Response:" << getReply->readAll();
        } else {
            qDebug() << "GET Error:" << getReply->errorString();
        }

        getReply->deleteLater();
    });

    // 进入应用程序事件循环
    return a.exec();
}

在使用 Qt 进行网络请求时,尤其是在进行异步的网络操作时,需要进入应用程序的事件循环。这是因为 Qt 的事件循环负责处理事件,而网络请求的完成(比如接收到服务器的响应)通常是通过 Qt 的信号-槽机制来处理的。

让我们来详细解释一下:

  1. 异步操作: Qt 的网络操作通常是异步的,即在发起网络请求后,程序会继续执行后续的代码而不等待请求完成。这是为了确保应用程序的界面和其他部分能够保持响应性,不被阻塞。

  2. 信号-槽机制: 当网络请求完成时,QNetworkReply 会发出 finished 信号。你在代码中使用 QObject::connect 来连接这个信号到一个槽函数,以便在请求完成时执行一些操作。

  3. 事件循环: 为了让信号-槽机制正常工作,需要进入应用程序的事件循环。调用 QCoreApplication::exec() 或者 QEventLoop::exec() 启动事件循环,使得 Qt 可以不断地检查并处理事件队列。

在代码中,调用 return a.exec(); 启动了事件循环。这样,当 GET 请求完成并发出 finished 信号时,相关的槽函数将会被执行。如果没有进入事件循环,这个槽函数将不会被触发,因为事件循环负责调度信号的处理。

简而言之,进入应用程序的事件循环是确保异步操作和信号-槽机制正常工作的关键步骤。如果你的应用程序没有事件循环,它将无法及时响应和处理异步操作的完成事件。

2.3 编程实现HTTP服务器

#ifndef MYHTTPSERVER_H
#define MYHTTPSERVER_H

#include <QTcpServer>
#include <QTcpSocket>
#include <QUrlQuery>
#include <QDebug>

class MyHTTPServer : public QTcpServer
{
    Q_OBJECT

public:
    MyHTTPServer(QObject *parent = nullptr) : QTcpServer(parent) {}

protected:
    //--------------------------------------
    // 说明:这是 QTcpServer 类的虚函数,当有新的连接到达时,会被调用。
    // 日期:2023-11-15
    // 作者:何浩文
    //--------------------------------------
    void incomingConnection(qintptr socketDescriptor) override
    {
        QTcpSocket *socket = new QTcpSocket(this);
        socket->setSocketDescriptor(socketDescriptor);

        // 读取客户端请求
        connect(socket, &QTcpSocket::readyRead, [&]() {
            QByteArray requestData = socket->readAll();
            processRequest(requestData, socket);

            // 关闭连接
            socket->disconnectFromHost();
        });

        // 处理连接断开
        connect(socket, &QTcpSocket::disconnected, [&]() {
            socket->deleteLater();
        });
    }

private:
    //--------------------------------------
    // 说明:这个函数用于解析 HTTP 请求,分析请求的方法和路径,并调用相应的处理函数。
    // 日期:2023-11-15
    // 作者:海码007
    //--------------------------------------
    void processRequest(const QByteArray &requestData, QTcpSocket *socket)
    {
        // 解析请求
        QString requestString = QString::fromUtf8(requestData);
        QStringList requestLines = requestString.split("\r\n");

        // 解析第一行,获取请求方法和路径
        QString firstLine = requestLines.first();
        QStringList parts = firstLine.split(" ");
        QString method = parts.value(0);
        QString path = parts.value(1);

        // 处理 GET 请求
        if (method == "GET")
        {
            handleGetRequest(path, socket);
        }
        // 处理 POST 请求
        else if (method == "POST")
        {
            handlePostRequest(path, requestData, socket);
        }
    }

    //--------------------------------------
    // 说明:处理 HTTP GET 请求的具体逻辑。
    // 日期:2023-11-15
    // 作者:海码007
    //--------------------------------------
    void handleGetRequest(const QString &path, QTcpSocket *socket)
    {
        QTextStream responseStream(socket);
        responseStream.setAutoDetectUnicode(true);

        // 构造HTTP响应
        responseStream << "HTTP/1.1 200 OK\r\n"
                       << "Content-Type: text/html\r\n"
                       << "Connection: close\r\n"
                       << "\r\n"
                       << "<html><body><h1>Hello, World! (GET)</h1></body></html>";

        // 刷新并等待数据发送完毕
        socket->flush();
        socket->waitForBytesWritten();
    }

    //--------------------------------------
    // 说明:处理 HTTP POST 请求的具体逻辑。
    // 日期:2023-11-15
    // 作者:海码007
    //--------------------------------------
    void handlePostRequest(const QString &path, const QByteArray &requestData, QTcpSocket *socket)
    {
        // 解析 POST 数据
        QUrlQuery postData(requestData);
        QString value = postData.queryItemValue("key");

        QTextStream responseStream(socket);
        responseStream.setAutoDetectUnicode(true);

        // 构造HTTP响应
        responseStream << "HTTP/1.1 200 OK\r\n"
                       << "Content-Type: text/html\r\n"
                       << "Connection: close\r\n"
                       << "\r\n"
                       << "<html><body><h1>Hello, " << value << "! (POST)</h1></body></html>";

        // 刷新并等待数据发送完毕
        socket->flush();
        socket->waitForBytesWritten();
    }
};

#endif // MYHTTPSERVER_H

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

FPGA时序分析与约束(13)——I/O接口约束

一、概述 在应用了时钟约束后&#xff0c;所有寄存器到寄存器的路径都能定时。为了获得更加精准的FPGA外部时序信息&#xff0c;设计者需要为FPGA的I/O接口指定时序信息&#xff0c;一般时序工具只能获取FPGA器件内部的时序信息&#xff0c;对于FPGA器件引脚之外的时序信息&…

Feature Pyramid Networks for Object Detection(2017.4)

文章目录 Abstract1. Introduction3. Feature Pyramid NetworksBottom-up pathwayTop-down pathway and lateral connections 7. Conclusion FPN Abstract 特征金字塔是识别系统中检测不同尺度物体的基本组成部分。但最近的深度学习对象检测器避免了金字塔表示&#xff0c;部分…

Swift的Copy on Write 简称CoW

了解Copy on Write在Swift开发时非常重要&#xff0c;因为这是Swift Standard Library的一个基础特性。 值类型&#xff1a;struct&#xff0c;enum&#xff0c;和tuple&#xff0c;比如在调用函数时传递参数&#xff0c;就会发送副本拷贝 引用类型&#xff1a;class&#xff…

用户的生命周期

用户生命周期是指用户在产品使用过程中的状态变化&#xff0c;一般分为5个阶段&#xff0c;分别为引入期、成长期、成熟期、沉默期和流失期。用户生命周期能够反映不同阶段用户的状态&#xff0c;可根据用户的不同状态进行针对性运营。运营中常说的拉新、促活、留存就是基于用户…

SLAM中提到的相机位姿到底指什么?

不小心又绕进去了&#xff0c;所以掰一下。 以我个人最直观的理解&#xff0c;假设无旋转&#xff0c;相机在世界坐标系的(5,0,0)^T的位置上&#xff0c;所谓“位姿”&#xff0c;应该反映相机的位置&#xff0c;所以相机位姿应该如下&#xff1a; Eigen::Matrix4d T Eigen::M…

上位机模块之halcon绘制ROI与获取ROI,在hsmartwindow实现

在上位机中通常需要使用到绘制ROI模块或者获取已经绘制好的ROI区域的参数&#xff0c;在这里通过使用hsmartwindow窗体控件进行对ROI的绘制和获取。 先上代码&#xff1a; /// <summary>/// 创建ROI/// </summary>/// <param name"Win">传入HSmar…

Centos7安装frps作内网穿透--实现外部访问家里群晖

实现在外可访问家用群晖 需要在外界访问家里的局域网设备&#xff0c;正常情况是需要有公网IP&#xff0c;而IPV4作为家用&#xff0c;运营商基本不给&#xff0c;除非钞能力&#xff0c;IPV6可以用&#xff0c;但是有缺陷&#xff0c;需要互访的两端都是IPV6才能访问。选择fr…

Mysql删除占用事务的线程

参考&#xff1a;https://www.jianshu.com/p/dd0291391188 产生原因&#xff1a;这个问题的原因是在mysql中产生了事务A&#xff0c;执行了修改的语句&#xff0c;比如&#xff1a; update t1 set aget18 where id1;此时事务并未进行提交&#xff0c;事务B开始运行&#xff0c…

kubernetes集群编排(10)

目录 prometheus监控 部署prometheus 部署nginx监控实例 部署prometheus-adapter prometheus监控 部署prometheus 创建项目仓库并上传镜像 [rootk8s2 helm]# vim prometheus-values.yaml alertmanager:alertmanagerSpec:image:repository: prometheus/alertmanagertag: v0.24.0…

知识解读:香港轻量云/云服务器/VPS性能差距解读

​  提起香港轻量云/云服务器/VPS 这三类&#xff0c;往往汇聚了中小企业和开发者等群体的讨论声音。当然&#xff0c;这跟它们本身产品定位有关&#xff0c;加上在初级配置这块价格上相差不大&#xff0c;也因此经常被拿来对比。 首先来简单了解一下最基础的区别&#xff1a…

Copliot:让你一秒变身网页达人的神奇助手

Copliot&#xff1a;一款能够帮助你快速理解网页内容的智能助手 你是否有过这样的经历&#xff0c;当你浏览网页时&#xff0c;遇到了一些你不太了解的内容&#xff0c;比如一些专业术语&#xff0c;一些复杂的概念&#xff0c;或者一些有趣的话题&#xff1f;你是否想要快速地…

Postman批量运行用例

近期在复习Postman的基础知识&#xff0c;在小破站上跟着百里老师系统复习了一遍&#xff0c;也做了一些笔记&#xff0c;希望可以给大家一点点启发。 一&#xff09;注意点 有上传文件的接口&#xff0c;需要做如下设置&#xff1a; 1、打开能读取外部文件的开关 2、把需要…

【IEEE】IF:10+, CCF|中科院1区TOP, 对国人友好,无需版面费

论文写作堪比西天取经&#xff0c;当我们经历“九九八十一难&#xff0c;取得真经“&#xff0c;还有最关键的一步&#xff0c;就是选刊发表。是“投石问路”&#xff0c;还是“投其所好”&#xff1f; 选刊有多重要&#xff0c;相信只要有过发表SCI经验的人都十分清楚。如果不…

登上CMMLU性能评测榜单第一 四大维度解码夸克自研大模型

11月14日&#xff0c;拥有千亿参数的夸克自研大模型正式发布&#xff0c;立刻占据CMMLU榜单第一名。夸克大模型将应用于通用搜索、医疗健康、教育学习、职场办公等多个场景。性能方面&#xff0c;其整体水平已经超过GPT-3.5&#xff0c;其中在写作、考试等部分场景中可以超过GP…

Android问题笔记四十五:解决SeekBar操作thumb周围产生的圆形阴影/灰色阴影/白色圆圈的解决办法

点击跳转专栏>Unity3D特效百例点击跳转专栏>案例项目实战源码点击跳转专栏>游戏脚本-辅助自动化点击跳转专栏>Android控件全解手册点击跳转专栏>Scratch编程案例点击跳转>软考全系列点击跳转>蓝桥系列点击跳转>ChatGPT和AIGC &#x1f449;关于作者 专…

线性回归模型用于波士顿房价预测的(普通VSsklearn库方法)比较

努力是为了不平庸~ 学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰 线性回归是一种统计学习方法&#xff0c;用于建立一个线性模型来预测因变量与自变量之间的关系。它假设因变量与自变量之间存在线性关系&#xff0c;…

移远EC600U-CN开发板 11.15

制作一个简单UI: 1."端口设置"模块 *效果图 *代码 def backEvent(evt): #返回主界面code evt.get_code() if code lv.EVENT.CLICKED:lv.scr_load(mainInterface)def popUpEvent(evt): #弹窗提醒code evt.get_code()if code lv.EVENT.CL…

YOLOv8改进实战 | 更换主干网络Backbone(五)之2023最新轻量化主干网络VanillaNet,深度学习中极简主义的力量

前言 轻量化网络设计是一种针对移动设备等资源受限环境的深度学习模型设计方法。下面是一些常见的轻量化网络设计方法: 网络剪枝:移除神经网络中冗余的连接和参数,以达到模型压缩和加速的目的。分组卷积:将卷积操作分解为若干个较小的卷积操作,并将它们分别作用于输入的不…

HackTheBox-Starting Point--Tier 2---Unified

文章目录 一 Unified 测试过程1.1 打点1.2 权限获取1.3 权限提升 二 题目 一 Unified 测试过程 1.1 打点 1.端口扫描 nmap -sV -sC 2.访问8080端口 页面跳转到&#xff1a;https://10.129.96.149:8443/manage/account/login?redirect%2Fmanage   观察到版本号为unifi 6.4.5…

【LeetCode刷题-滑动窗口】--1423.可获得的最大点数

1423.可获得的最大点数 思路&#xff1a; 数组cardPoints的长度为n&#xff0c;由于只能从开头和末尾拿k张卡牌&#xff0c;所以最后剩下的必然是连续的n-k张卡牌&#xff0c;可以通过求出剩余卡牌点数之和的最小值&#xff0c;来求出拿走卡牌点数之和的最大值 算法&#xff…