最新资讯

  • 【高并发服务器:HTTP应用】十四、Util工具类的设计与实现

【高并发服务器:HTTP应用】十四、Util工具类的设计与实现

2026-01-28 15:58:32 栏目:最新资讯 2 阅读


文章目录

  • Ⅰ. `HTTP`协议模块的子模块划分
    • 1、`Util`工具模块
    • 2、`HttpRequest`请求模块
    • 3、`HttpResponse`响应模块
    • 4、`HttpContext`上下文模块
    • 5、`HttpServer`服务器模块
  • Ⅱ. `Util`工具类功能设计与类设计
  • Ⅲ. 接口实现
    • 1、字符串分割函数 `split()`
    • 2、从文件读取数据 `read_file()`
    • 3、向文件写入数据 `write_file()`
    • 4、`URL`编码 `url_encode()`
    • 5、`URL`解码 `url_decode()`
    • 6、状态码对应信息 && 请求资源格式
    • 7、判断文件是否为一个目录/普通文件
    • 8、判断一个`HTTP`资源路径是否有效

Ⅰ. HTTP协议模块的子模块划分

​ 我们实现了 muduo 服务器框架之后,就可以被很多应用层协议包装起来使用,下面我们就以最常见的 HTTP 协议为例,搭建一个 HTTP 服务器!

HTTP 协议模块用于对高并发服务器模块进行协议支持,基于提供的协议支持能够更方便的完成指定协议服务器的搭建。而 HTTP 协议支持模块的实现,可以细分为以下几个模块。

1、Util工具模块

​ 这个模块的功能就是 提供一些服务器需要的功能性工具接口,方便使用者直接调用!大概需要提供以下这些接口:

  1. 读取文件内容
  2. 向文件写入内容
  3. URL 编码
  4. URL 解码
  5. 通过 HTTP 状态码获取描述信息
  6. 根据文件后缀名获取 mime(即想要请求的文件格式)
  7. 判断一个文件是否是目录
  8. 判断一个文件是否是一个普通文件
  9. 判断一个 HTTP 资源路径是否有效

2、HttpRequest请求模块

​ 这个模块是 HTTP 请求数据模块,用于存储 HTTP 的请求信息,然后按照 HTTP 请求格式进行解析,得到各个关键要素放到 Request 中,这样子能让 HTTP 请求的分析更加的方便!

​ 要分析的内容无非就是请求信息,如下所示:

  • 请求方法
  • URL
    • 资源路径
    • 查询字符串
  • 协议版本
  • 头部字段
  • 正文

所以需要提供以下接口:

  1. 头部字段的插入和获取
  2. 查询字符串的插入和获取
  3. 长连接和短连接的判断
  4. 正文长度的获取

3、HttpResponse响应模块

​ 这个模块和上面的请求模块就是相反的,主要存储 HTTP 的响应信息,在进行业务处理的同时,让使用者向 Response 中填充响应要素,完毕后将其组织成为 HTTP 响应格式的数据,发送给客户端,这样子能让 HTTP 响应的过程操作变得简单!

​ 要操作的内容无非就是响应信息,如下所示:

  • 协议版本
  • 响应状态码
  • 状态码描述信息
  • 头部字段
  • 正文

所以需要提供以下接口:

  1. 头部字段的插入和获取
  2. 长连接和短连接的设置与判断
  3. 正文的设置

4、HttpContext上下文模块

​ 这个模块用于记录 HTTP 请求的接收和处理进度,因为有可能出现一种情况,就是接收的数据并不是一条完整的 HTTP 请求数据,也就是请求的处理需要在多次收到数据后才能处理完成,所以在每次处理的时候,就需要将处理进度记录起来,以便于下次从当前进度继续向下处理,最终得到一个完整 HttpRequest 请求信息对象,因此 在请求数据的接收以及解析部分需要一个上下文来进行控制接收和处理节奏

​ 这个模块要关心的要素如下所示:

  • 已经接收并处理的请求信息
  • 接收状态(即处于何种阶段):
    • 接收请求行(当前处于接收并处理请求行的阶段)
    • 接收请求头部(表示请求头部的接收还没有完毕)
    • 接收正文(表示还有正文没有接收完毕)
    • 接收数据完毕(这是一个接收完毕,可以对请求进行处理的阶段)
    • 接收处理请求出错
  • 响应状态码:
    • 在请求的接收并处理中,有可能会出现各种不同的问题,比如解析出错、访问的资源不对、没有权限等等,而这些错误的响应状态码都是不一样的。

所以需要提供以下接口:

  1. 接收并处理请求数据:
    • 接收请求行
    • 解析请求行
    • 接收头部
    • 解析头部
    • 接收正文
  2. 返回解析完毕的请求信息
  3. 返回响应状态码
  4. 返回接收解析状态

5、HttpServer服务器模块

​ 这个模块是最终给组件使用者提供的 HTTP 服务器模块了,用于以简单的接口实现 HTTP 服务器的搭建。

HttpServer 模块内部如下元素:

  • 一个 TcpServer 对象:实现服务器的搭建。
  • 两个提供给 TcpServer 对象的接口:连接建立成功设置上下文接口,数据处理接口。
  • 一个哈希表存储请求与处理函数的映射表:组件使用者向 HttpServer 设置哪些请求应该使用哪些函数进行处理,等 TcpServer 收到对应的请求就会使用对应的函数进行处理。

Ⅱ. Util工具类功能设计与类设计

这个模块的功能就是 提供一些服务器需要的功能性工具接口,方便使用者直接调用!大概需要提供以下这些接口:

  1. 从文件读取内容
  2. 向文件写入内容
  3. URL 编码
  4. URL 解码
  5. 通过 HTTP 状态码获取描述信息
  6. 根据文件后缀名获取 mime(即想要请求的文件格式)
  7. 判断一个文件是否是目录
  8. 判断一个文件是否是一个普通文件
  9. 判断一个 HTTP 资源路径是否有效

​ 下面是 Util 类的大体框架:(实现时候会有其它辅助函数)

class Util
{
public:
    // 字符串分割函数,将src字符串按照sep字符串进行分割,得到的各个子串放到arr中,最终返回子串的数量
    static size_t split(const std::string& src, const std::string& sep, std::vector<std::string>* arr);
    
    // 从文件读取内容,将读取的内容放到一个buffer中
    static bool read_file(const std::string& filename, std::string* buffer);

    // 向文件写入内容,要写的内容存放在buffer中
    static bool write_file(const std::string& filename, const std::string& buffer);

    // URL编码,避免URL中资源路径与查询字符串中的特殊字符与HTTP请求中特殊字符产生歧义
    // 编码格式:将特殊字符的ascii值,转换为两个16进制字符以及一个前缀%   比如C++ -> C%2B%2B
    //   不编码的特殊字符: RFC3986文档规定 . - _ ~ 字母,数字属于绝对不编码字符
    // RFC3986文档规定,编码格式 %HH 
    // W3C标准中规定,查询字符串中的空格,需要编码为+, 解码则是+转空格
    static std::string url_encode(const std::string& url, bool convert_space_to_plus);

    // URL解码
    static std::string url_decode(const std::string& url, bool convert_plus_to_space);

    // 通过HTTP状态码获取描述信息
    static std::string get_information_from_status(int status);

    // 根据文件后缀名获取mime(即想要请求的文件格式)
    static std::string get_mime_from_suffix(const std::string& filename);

    // 判断一个文件是否是目录
    static bool is_directory(const std::string& filename);

    // 判断一个文件是否是一个普通文件
    static bool is_regular_file(const std::string& filename);

    // 判断一个HTTP资源路径是否有效
    //      /index.html  --- 前边的/叫做相对根目录,映射的是某个服务器上的子目录
    //      想表达的意思就是,客户端只能请求相对根目录中的资源,其他地方的资源都不予理会
    //      而/../login --- 这个路径中的..会让路径的查找跑到相对根目录之外,这是不合理的,不安全的
    static bool is_path_valid(const std::string& path);
};

Ⅲ. 接口实现

1、字符串分割函数 split()

​ 实现并不难,但是要注意细节,就是如果有重复的 sep 字符串的话,此时我们需要判断一下 curpre 的关系,如果它们截取的是一个空串的话,那么就得需要过滤一下,简单地说就是如果截取长度为零的话就不用处理!

// 字符串分割函数,将src字符串按照sep字符串进行分割,得到的各个子串放到arr中,最终返回子串的数量
static size_t split(const std::string& src, const std::string& sep, std::vector<std::string>* arr)
{
    // src:"There are two needles in this haystack with needles."
    // sep:"needle"
    // 则第一个位置为14
    size_t cur = src.find(sep);
    size_t pre = 0;
    while(cur != std::string::npos)
    {
        if(cur - pre > 0)
            arr->push_back(src.substr(pre, cur - pre)); // 要过滤掉空串的情况
        pre = cur + sep.size();
        cur = src.find(sep, cur + sep.size());
    }   
    if(src.size() - pre > 0)
        arr->push_back(src.substr(pre)); // 别忘了最后一个子串(也要判断是否为空串)

    return arr->size();
}

​ 测试用例:

#include "httpserver.hpp"

int main()
{
    // std::string src = "There are two needles in this haystack with needles.";
    // std::string sep = "needle";
    std::string src = "abc asd    asdv ";
    std::string sep = " ";
    std::vector<std::string> arr;
    size_t n = Util::split(src, sep, &arr);
    for(int i = 0; i < n; ++i)
        DLOG("[%s]", arr[i].c_str());
    return 0;
}

[liren@VM-8-7-centos http]$ ./main 
[140516807022400 2023-10-30 15:02:45 main.cc:12] [abc]
[140516807022400 2023-10-30 15:02:45 main.cc:12] [asd]
[140516807022400 2023-10-30 15:02:45 main.cc:12] [asdv]

2、从文件读取数据 read_file()

​ 这里我们使用的是 c++ 中的输入文件流 std::ifstream 来操作,具体参考下面代码:

// 从文件读取内容,将读取的内容放到一个buffer中
static bool read_file(const std::string& filename, std::string* buffer)
{
    // 1. 创建二进制输入文件流
    std::ifstream ifs(filename, std::ios::binary);
    if(ifs.is_open() == false)
    {
        ELOG("open %s file failed!", filename.c_str());
        return false;
    }

    // 2. 偏移到末尾,然后通过此时的偏移量来获取文件的大小,最后别忘了重新偏移到开头
    ifs.seekg(0, ifs.end); 
    size_t file_size = ifs.tellg();
    ifs.seekg(0, ifs.beg);

    // 3. 根据文件大小写到buffer中
    buffer->resize(file_size);
    ifs.read(&(*buffer)[0], file_size); // 因为c_str()返回的是常量字符串,和参数不匹配,所以这里只能用取地址的方式传入buffer
    if(ifs.good() == false)
    {
        ELOG("read %s file failed!", filename.c_str());
        ifs.close();
        return false;
    }

    // 4. 最后关闭文件流
    ifs.close();
    return true;
}

​ 测试代码(执行结果只给出部分,节省篇幅):

#include "httpserver.hpp"

int main()
{
    std::string buffer;
    bool ret = Util::read_file("../../example/eventfd.cpp", &buffer);
    if(ret != false)
        DLOG("
%s", buffer.c_str());
    return 0;
}

// 执行结果:
[liren@VM-8-7-centos http]$ ./main 
[140639060961088 2023-10-30 15:22:26 main.cc:8] 
#include 
#include 
#include 
using namespace std;
int main()
{
    // 创建一个eventfd
    int efd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
    if(efd < 0)
    {
        perror("eventfd error");
……

3、向文件写入数据 write_file()

// 向文件写入内容,要写的内容存放在buffer中
static bool write_file(const std::string& filename, const std::string& buffer)
{
    // 1. 创建二进制输出文件流
    std::ofstream ofs(filename, std::ios::binary);
    if(ofs.is_open() == false)
    {
        ELOG("open %s file failed!", filename.c_str());
        return false;
    }

    // 2. 将buffer中的数据写入文件中
    ofs.write(buffer.c_str(), buffer.size());
    if(ofs.good() == false)
    {
        ELOG("write %s file failed!", filename.c_str());
        ofs.close();
        return false;
    }

    // 3. 最后关闭文件流
    ofs.close();
    return true;
}

​ 测试用例(无非就是上面读取完的结果,写入到一个新文件,然后看看两个文件的 md5 值的区别):

#include "httpserver.hpp"

int main()
{
    std::string buffer;
    bool ret = Util::read_file("../../example/eventfd.cpp", &buffer);
    if(ret != false)
        DLOG("
%s", buffer.c_str());
    
    // 将buffer中的数据写入新文件
    ret = Util::write_file("../../test/write_test.txt", buffer);
    if(ret != false)
        DLOG("
%s", buffer.c_str());
    return 0;
}

4、URL编码 url_encode()

​ 编码其实不难,主要是注意编码的规则要求,就是将特殊字符的 ascii 值,转换为两个十六进制字符,然后前面带上 % 即可,也就是 %HH 的格式!不过还有一些不编码的特殊字符,比如 RFC3986文档规定 .-_~、字母以及数字属于绝对不编码字符!

​ 此外 W3C 标准中规定,查询字符串中的空格,需要编码为 +,而解码则是 + 转空格。所以为了使用者方便,我们提供一个布尔值变量 convert_space_to_plus 由使用者设置,如果想让空格变成 + 号的话,则让其为 true 即可!

// URL编码,避免URL中资源路径与查询字符串中的特殊字符与HTTP请求中特殊字符产生歧义
static std::string url_encode(const std::string& url, bool convert_space_to_plus)
{
    std::string ret;
    for(int i = 0; i < url.size(); ++i)
    {
        // 数字、字母以及不编码的字符直接尾插即可
        if(url[i] == '.' || url[i] == '-' || url[i] == '_' || url[i] == '~' || isalnum(url[i])) 
            ret += url[i];
        else if(url[i] == ' ' && convert_space_to_plus == true) // 如果为空格并且要求转化为+而不是编码的话,则直接尾插+即可
            ret += '+';
        else
        {
            // 剩下的就是要编码的符号了!
            // 下面直接使用 snprintf() 函数,格式化然后存放到tmp中,最后再尾插到ret即可
            char tmp[4] = { 0 };
            snprintf(tmp, 4, "%%%02X", url[i]); // %%表示输出真实的%,%02X表示输出占2位,不够位的话补充前置0的大写十六进制数
            ret += tmp;
        }
    }
    return ret;
}

​ 测试代码:

#include "httpserver.hpp"

int main()
{
    std::cout << Util::url_encode("c++", false) << std::endl;

    std::cout << Util::url_encode("c  ", true) << std::endl;
    std::cout << Util::url_encode("c  ", false) << std::endl;

    std::cout << Util::url_encode("/login/user=lirendada & id = 2", true) << std::endl;
    std::cout << Util::url_encode("/login/user=lirendada & id = 2", false) << std::endl;
    return 0;
}

// 执行结果:
[liren@VM-8-7-centos http]$ ./main 
c%2B%2B
c++
c%20%20
%2Flogin%2Fuser%3Dlirendada+%26+id+%3D+2
%2Flogin%2Fuser%3Dlirendada%20%26%20id%20%3D%202

5、URL解码 url_decode()

​ 解码比编码其实要简单一些,就是遇到 % 号然后对其后面两个十六进制位进行转化后变成十进制位即可,其中第一个数字左移 4 位(也就是乘以 16)然后加上第二个数字,比如 %2b 变成 (2<<4)+11 = 43

​ 其中将十六进制转化为十进制的功能函数提炼出来,单独作为一个模块 hex_to_dec()

// URL解码
static std::string url_decode(const std::string& url, bool convert_plus_to_space)
{   
    // 即遇到了%号则将紧随其后的2个字符转换为数字,第一个数字左移4位(也就是乘以16)然后加上第二个数字,比如%2b -> 2<<4+11 = 43
    std::string ret;
    for(int i = 0; i < url.size(); ++i)
    {
        // 对于+号的特殊处理
        if(url[i] == '+' && convert_plus_to_space == true)
        {
            ret += ' ';
            continue;
        }

        if(url[i] == '%' && (i + 2) < url.size())
        {
            int n1 = hex_to_dec(url[i + 1]);
            int n2 = hex_to_dec(url[i + 2]);
            ret += (char)((n1 << 4) + n2);
            i += 2;
            continue;
        }

        // 剩下的就是不需要解码的
        ret += url[i];
    }
    return ret;
}

// 十六进制转十进制的功能函数
static int hex_to_dec(char c)
{
    if(c >= '0' && c <= '9')
        return c - '0';
    else if(c >= 'a' && c <= 'z')
        return c - 'a' + 10;
    else if(c >= 'A' && c <= 'Z')
        return c - 'A' + 10;
    return -1;
}

​ 测试代码:

#include "httpserver.hpp"

int main()
{
    std::cout << Util::url_decode("c++", true) << std::endl;
    std::cout << Util::url_decode("c++", false) << std::endl << std::endl;

    std::cout << Util::url_decode("c%20%20", true) << std::endl;
    std::cout << Util::url_decode("c%20%20", false) << std::endl << std::endl;

    std::cout << Util::url_decode("%2Flogin%2Fuser%3Dlirendada+%26+id+%3D+2", false) << std::endl;
    std::cout << Util::url_decode("%2Flogin%2Fuser%3Dlirendada%20%26%20id%20%3D%202", false) << std::endl;
    return 0;
}

// 执行结果:
[liren@VM-8-7-centos http]$ ./main 
c  
c++
   
c  
c  

/login/user=lirendada+&+id+=+2
/login/user=lirendada & id = 2

6、状态码对应信息 && 请求资源格式

​ 对应这两个文件,其实可以直接从网上找到,这里我们将其作为全局变量放到哈希表中使用即可!

// 状态码对应信息的哈希表
std::unordered_map<int, std::string> status_msg = {
    {100,  "Continue"},
    {101,  "Switching Protocol"},
    {102,  "Processing"},
    {103,  "Early Hints"},
    {200,  "OK"},
    {201,  "Created"},
    {202,  "Accepted"},
    {203,  "Non-Authoritative Information"},
    {204,  "No Content"},
    {205,  "Reset Content"},
    {206,  "Partial Content"},
    {207,  "Multi-Status"},
    {208,  "Already Reported"},
    {226,  "IM Used"},
    {300,  "Multiple Choice"},
    {301,  "Moved Permanently"},
    {302,  "Found"},
    {303,  "See Other"},
    {304,  "Not Modified"},
    {305,  "Use Proxy"},
    {306,  "unused"},
    {307,  "Temporary Redirect"},
    {308,  "Permanent Redirect"},
    {400,  "Bad Request"},
    {401,  "Unauthorized"},
    {402,  "Payment Required"},
    {403,  "Forbidden"},
    {404,  "Not Found"},
    {405,  "Method Not Allowed"},
    {406,  "Not Acceptable"},
    {407,  "Proxy Authentication Required"},
    {408,  "Request Timeout"},
    {409,  "Conflict"},
    {410,  "Gone"},
    {411,  "Length Required"},
    {412,  "Precondition Failed"},
    {413,  "Payload Too Large"},
    {414,  "URI Too Long"},
    {415,  "Unsupported Media Type"},
    {416,  "Range Not Satisfiable"},
    {417,  "Expectation Failed"},
    {418,  "I'm a teapot"},
    {421,  "Misdirected Request"},
    {422,  "Unprocessable Entity"},
    {423,  "Locked"},
    {424,  "Failed Dependency"},
    {425,  "Too Early"},
    {426,  "Upgrade Required"},
    {428,  "Precondition Required"},
    {429,  "Too Many Requests"},
    {431,  "Request Header Fields Too Large"},
    {451,  "Unavailable For Legal Reasons"},
    {501,  "Not Implemented"},
    {502,  "Bad Gateway"},
    {503,  "Service Unavailable"},
    {504,  "Gateway Timeout"},
    {505,  "HTTP Version Not Supported"},
    {506,  "Variant Also Negotiates"},
    {507,  "Insufficient Storage"},
    {508,  "Loop Detected"},
    {510,  "Not Extended"},
    {511,  "Network Authentication Required"}
};

// 文件后缀对应资源格式的哈希表
std::unordered_map<std::string, std::string> mime_msg = {
    {".aac",        "audio/aac"},
    {".abw",        "application/x-abiword"},
    {".arc",        "application/x-freearc"},
    {".avi",        "video/x-msvideo"},
    {".azw",        "application/vnd.amazon.ebook"},
    {".bin",        "application/octet-stream"},
    {".bmp",        "image/bmp"},
    {".bz",         "application/x-bzip"},
    {".bz2",        "application/x-bzip2"},
    {".csh",        "application/x-csh"},
    {".css",        "text/css"},
    {".csv",        "text/csv"},
    {".doc",        "application/msword"},
    {".docx",       "application/vnd.openxmlformats-officedocument.wordprocessingml.document"},
    {".eot",        "application/vnd.ms-fontobject"},
    {".epub",       "application/epub+zip"},
    {".gif",        "image/gif"},
    {".htm",        "text/html"},
    {".html",       "text/html"},
    {".ico",        "image/vnd.microsoft.icon"},
    {".ics",        "text/calendar"},
    {".jar",        "application/java-archive"},
    {".jpeg",       "image/jpeg"},
    {".jpg",        "image/jpeg"},
    {".js",         "text/javascript"},
    {".json",       "application/json"},
    {".jsonld",     "application/ld+json"},
    {".mid",        "audio/midi"},
    {".midi",       "audio/x-midi"},
    {".mjs",        "text/javascript"},
    {".mp3",        "audio/mpeg"},
    {".mpeg",       "video/mpeg"},
    {".mpkg",       "application/vnd.apple.installer+xml"},
    {".odp",        "application/vnd.oasis.opendocument.presentation"},
    {".ods",        "application/vnd.oasis.opendocument.spreadsheet"},
    {".odt",        "application/vnd.oasis.opendocument.text"},
    {".oga",        "audio/ogg"},
    {".ogv",        "video/ogg"},
    {".ogx",        "application/ogg"},
    {".otf",        "font/otf"},
    {".png",        "image/png"},
    {".pdf",        "application/pdf"},
    {".ppt",        "application/vnd.ms-powerpoint"},
    {".pptx",       "application/vnd.openxmlformats-officedocument.presentationml.presentation"},
    {".rar",        "application/x-rar-compressed"},
    {".rtf",        "application/rtf"},
    {".sh",         "application/x-sh"},
    {".svg",        "image/svg+xml"},
    {".swf",        "application/x-shockwave-flash"},
    {".tar",        "application/x-tar"},
    {".tif",        "image/tiff"},
    {".tiff",       "image/tiff"},
    {".ttf",        "font/ttf"},
    {".txt",        "text/plain"},
    {".vsd",        "application/vnd.visio"},
    {".wav",        "audio/wav"},
    {".weba",       "audio/webm"},
    {".webm",       "video/webm"},
    {".webp",       "image/webp"},
    {".woff",       "font/woff"},
    {".woff2",      "font/woff2"},
    {".xhtml",      "application/xhtml+xml"},
    {".xls",        "application/vnd.ms-excel"},
    {".xlsx",       "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"},
    {".xml",        "application/xml"},
    {".xul",        "application/vnd.mozilla.xul+xml"},
    {".zip",        "application/zip"},
    {".3gp",        "video/3gpp"},
    {".3g2",        "video/3gpp2"},
    {".7z",         "application/x-7z-compressed"}
};

​ 有个这两个哈希表之后,我们的工具函数实现起来就非常的简单了,就是哈希表的基本操作,如下所示:

// 通过HTTP状态码获取描述信息
static std::string get_information_from_status(int status) 
{ 
    auto it = status_msg.find(status);
    if(it == status_msg.end())
        return "unknown status";
    return status_msg[status];
}

// 根据文件后缀名获取mime(即想要请求的文件格式)
static std::string get_mime_from_suffix(const std::string& filename) 
{ 
    // 先获取后缀名
    size_t pos = filename.find_last_of('.');
    if(pos == std::string::npos)
        return "application/octet-stream";     // 错误或者没找到的话返回一个二进制流的格式
    std::string suffix = filename.substr(pos);

    // 再判断存不存在
    auto it = mime_msg.find(suffix);
    if(it == mime_msg.end())
        return "application/octet-stream";     // 错误或者没找到的话返回一个二进制流的格式
    return mime_msg[suffix];
}

7、判断文件是否为一个目录/普通文件

​ 如果我们直接用字符串操作去判断的话不太好搞,我们可以使用 stat() 函数,获取文件的属性,然后通过其中的 st_mode 也就是文件的类型,辅助两个宏,一个是 S_ISDIR,另一个是 S_ISREG,分别判断是否为目录和普通文件类型即可!

// 判断一个文件是否是目录
static bool is_directory(const std::string& filename)
{
    // 通过stat函数获取文件的属性,然后通过S_ISDIR来判断其中的文件类型是否为目录即可
    struct stat buf = { 0 };
    int ret = stat(filename.c_str(), &buf);
    if(ret == -1)
        return false;

    return S_ISDIR(buf.st_mode);
}

// 判断一个文件是否是一个普通文件
static bool is_regular_file(const std::string& filename)
{
    // 通过stat函数获取文件的属性,然后通过S_ISREG来判断其中的文件类型是否为目录即可
    struct stat buf = { 0 };
    int ret = stat(filename.c_str(), &buf);
    if(ret == -1)
        return false;

    return S_ISREG(buf.st_mode);
}

8、判断一个HTTP资源路径是否有效

​ 判断路径是否有效,其实只需要判断一下该路径是否非法访问当前根目录以外的资源即可!

​ 我们可以以斜杆 / 为分割字符,进行字符串分割,此时分割出多少个子串,就相当于进入了多少层目录,但是因为有 .. 的存在,我们就需要遍历判断一下这些子串有多少个 ..,然后用一个变量 level 记录当前的层数大小,遇到 .. 则让 level--,表示退出该层目录;如果不是的话则让 level++ 表示进入了一层目录!然后只要 判断 level 是否小于零即可,如果小于零说明访问到了根目录以外的内容,是不合法的,直接返回错误即可!

// 判断一个HTTP资源路径是否有效
//      /index.html  --- 前边的/叫做相对根目录,映射的是某个服务器上的子目录
//      想表达的意思就是,客户端只能请求相对根目录中的资源,其他地方的资源都不予理会
//      而/../login --- 这个路径中的..会让路径的查找跑到相对根目录之外,这是不合理的,不安全的
static bool is_path_valid(const std::string& path)
{
    // 1. 以斜杆/为分割字符,进行字符串分割
    std::vector<std::string> substr;
    split(path, "/", &substr);

    // 2. 然后遍历每个子串
    int level = 0; // 表示当前的层数
    for(int i = 0; i < substr.size(); ++i)
    {
        if(substr[i] == "..")
        {
            level--;
            if(level < 0)
                return false; // 如果小于0说明访问到了根目录以外的内容,是不合法的,直接false
        }
        else
            level++; // 如果不是..的话,那么有多少个子串,就相当于进入了多少个目录,则让level++即可
    }
    return true;
}

本文地址:https://www.yitenyun.com/440.html

搜索文章

Tags

#ios面试 #ios弱网 #断点续传 #ios开发 #objective-c #ios #ios缓存 #服务器 #python #pip #conda #远程工作 香港站群服务器 多IP服务器 香港站群 站群服务器 #kubernetes #笔记 #平面 #容器 #linux #学习方法 #运维 #fastapi #html #css #进程控制 #docker #后端 #数据库 #低代码 #爬虫 #音视频 #cpolar #Conda # 私有索引 # 包管理 #内网穿透 #网络 #开发语言 #云原生 #iventoy #VmWare #OpenEuler #人工智能 #node.js #MobaXterm #ubuntu #开源 #RTP over RTSP #RTP over TCP #RTSP服务器 #RTP #TCP发送RTP #android #腾讯云 #c# #Trae #IDE #AI 原生集成开发环境 #Trae AI #tcp/ip #多个客户端访问 #IO多路复用 #回显服务器 #TCP相关API #web安全 #学习 #安全 #kylin #数信院生信服务器 #Rstudio #生信入门 #生信云服务器 #物联网 #websocket #vscode #mobaxterm #深度学习 #计算机视觉 #华为 #ModelEngine #github #git #金融 #大模型 #mcp #金融投资Agent #Agent #n8n #本地部署 #windows #我的世界 #hadoop #hbase #hive #zookeeper #spark #kafka #flink #qt #C++ #我的世界服务器搭建 #minecraft #云计算 #java #jar #nginx #claude #阿里云 #ide #AI编程 #架构 #mvp #个人开发 #设计模式 #SRS #流媒体 #直播 #DisM++ # GLM-4.6V # 系统维护 #京东云 #gpu算力 #算法 #守护进程 #复用 #screen #umeditor粘贴word #ueditor粘贴word #ueditor复制word #ueditor上传word图片 #缓存 #unity3d #游戏 #服务器框架 #Fantasy #http #c++ #性能优化 #mamba #ssh #智能手机 #MCP #科技 #自然语言处理 #神经网络 #todesk #udp #c语言 #网络协议 #unity #游戏引擎 #jenkins #MCP服务器 #需求分析 #scala #测试用例 #测试工具 #压力测试 #Dell #PowerEdge620 #内存 #硬盘 #RAID5 #面试 #NPU #CANN #vue.js #前端 #vue #gemini #gemini国内访问 #gemini api #gemini中转搭建 #Cloudflare #JumpServer #堡垒机 #stm32 #macos #振镜 #振镜焊接 #pycharm #单元测试 #集成测试 #嵌入式硬件 #编辑器 #SAP #ebs #metaerp #oracle ebs #AIGC #ida #大数据 #搜索引擎 #DeepSeek #蓝耘智算 #pytorch #Nacos #web #微服务 #Anaconda配置云虚拟环境 #elasticsearch #YOLOFuse # Base64编码 # 多模态检测 #django #flask #web3.py #oracle #RustDesk #IndexTTS 2.0 #本地化部署 #麒麟OS #centos #毕业设计 #车辆排放 #PyTorch # Triton # 高并发部署 #ms-swift # 大模型 # 模型训练 #Android #Bluedroid #java-ee #AI #工具集 #mysql #sql #jmeter #功能测试 #软件测试 #自动化测试 #职场和发展 #Dify #ARM架构 #鲲鹏 #EMC存储 #存储维护 #NetApp存储 #910B #golang #rdp #libosinfo #自动化 #maven #gitlab #NAS #Termux #Samba #Linux #SSH反向隧道 # Miniconda # Jupyter远程访问 #chatgpt #codex #三维 #3D #三维重建 #单片机 #php #apache #鸭科夫 #逃离鸭科夫 #鸭科夫联机 #鸭科夫异地联机 #开服 #deepseek #risc-v #GPU服务器 #8U #硬件架构 #cpp #项目 #高并发 #智能路由器 #5G #C2000 #TI #实时控制MCU #AI服务器电源 #1024程序员节 #运维开发 #chrome #黑群晖 #虚拟机 #无U盘 #纯小白 #支付 #银河麒麟 #系统升级 #信创 #国产化 #SSH跳板机 # Python3.11 #fpga开发 #LVDS #高速ADC #DDR #东方仙盟 #API限流 # 频率限制 # 令牌桶算法 #驱动开发 #YOLO # 目标检测 #经验分享 #华为云 #YOLO26 #目标检测 #screen命令 #机器人 #Gunicorn #WSGI #Flask #并发模型 #容器化 #Python #性能调优 #蓝湖 #Axure原型发布 #milvus #课程设计 #springboot #ai #知识库 #web server #请求处理流程 #llama #opencv #语言模型 #spring boot #门禁 #梯控 #智能一卡通 #门禁一卡通 #消费一卡通 #智能梯控 #一卡通 #源代码管理 #小程序 #微信小程序 #超时设置 #客户端/服务器 #网络编程 #C# #数据结构 #管道Pipe #system V #ai编程 #交通物流 #muduo库 #javascript #uv #uvx #uv pip #npx #Ruff #pytest #react native #react.js #昇腾 #rocketmq #selenium #操作系统 #国产化OS #scrapy #蓝牙 #LE Audio #BAP #C语言 #vivado license #CVE-2025-68143 #CVE-2025-68144 #CVE-2025-68145 #网络安全 #嵌入式编译 #ccache #distcc #SSH # 批量管理 #语音识别 #ASR #SenseVoice #星图GPU #中间件 #MQTT协议 #智能体 #openHiTLS #TLCP #DTLCP #密码学 #商用密码算法 #ONLYOFFICE #MCP 服务器 #laravel #ssl #tomcat #链表 #puppeteer #asp.net #prometheus #grafana #svn #证书 #可信计算技术 #serverless #RAID #RAID技术 #磁盘 #存储 #dlms #dlms协议 #逻辑设备 #逻辑设置间权限 #bug菌问答团队 #测评 #CCE #Dify-LLM #Flexus #安全威胁分析 #仙盟创梦IDE # 双因素认证 # TensorFlow #动态规划 #负载均衡 #服务器繁忙 #xlwings #Excel #nfs #iscsi #postgresql #连接数据库报错 #cursor #jvm #adb #spring cloud #spring #transformer #prompt #大模型学习 #文件管理 #文件服务器 #硬件工程 #智能家居 #pyqt #DNS #SPA #单页应用 #redis #C #树莓派4b安装系统 #Spring AI #STDIO传输 #SSE传输 #WebMVC #WebFlux #bootstrap #电脑 #信息与通信 #信号处理 #tcpdump #swagger #visual studio code #排序算法 #jdk #排序 #ddos #系统架构 #debian #mariadb #分布式 #KMS激活 #计算机网络 #LangGraph #CLI #JavaScript #langgraph.json #数据仓库 #paddleocr #CSDN #wsl #银河麒麟高级服务器操作系统安装 #银河麒麟高级服务器V11配置 #设置基础软件仓库时出错 #银河麒高级服务器系统的实操教程 #生产级部署银河麒麟服务系统教程 #Linux系统的快速上手教程 #微信 #sqlite #intellij-idea #numpy #epoll #电气工程 #PLC #wordpress #雨云 #LobeChat #vLLM #GPU加速 #Puppet # IndexTTS2 # TTS #海外服务器安装宝塔面板 #翻译 #开源工具 #ansible #机器学习 #说话人验证 #声纹识别 #CAM++ #大模型部署 #mindie #大模型推理 #创业创新 #业界资讯 #openlayers #bmap #tile #server #langchain #大模型开发 #程序员 #TCP #客户端 #嵌入式 #DIY机器人工房 #CosyVoice3 # 语音合成 #unix #rust #x86_64 #数字人系统 #yum #json #windows11 #microsoft #系统修复 #制造 #信令服务器 #Janus #MediaSoup #ping通服务器 #读不了内网数据库 #其他 #rtsp #转发 #大模型教程 #AI大模型 #结构体 #渗透测试 #黑客技术 #计算机 #文件上传漏洞 #mcu #CVE-2025-61686 #漏洞 #路径遍历高危漏洞 #数据分析 #推荐算法 #SQL注入主机 #shell #SSE # AI翻译机 # 实时翻译 #webrtc #idm #万悟 #联通元景 #镜像 #A2A #GenAI #ThingsBoard MCP #聊天小程序 # GPU租赁 # 自建服务器 #心理健康服务平台 #心理健康系统 #心理服务平台 #心理健康小程序 #devops #戴尔服务器 #戴尔730 #装系统 #健身房预约系统 #健身房管理系统 #健身管理系统 #bug #sqlserver #nodejs #HeyGem # 服务器IP访问 # 端口映射 #遛狗 #esb接口 #走处理类报异常 #ffmpeg #数据挖掘 #交互 #北京百思可瑞教育 #百思可瑞教育 #北京百思教育 #练习 #基础练习 #数组 #循环 #九九乘法表 #计算机实现 #dynadot #域名 #银河麒麟部署 #银河麒麟部署文档 #银河麒麟linux #银河麒麟linux部署教程 # 一锤定音 # 大模型微调 #数据安全 #注入漏洞 #vllm #CUDA #Triton #SSH公钥认证 # PyTorch # 安全加固 #leetcode #dify #AI 推理 #NV #部署 #企业开发 #ERP #项目实践 #.NET开发 #C#编程 #编程与数学 #昇腾300I DUO #fiddler #vnstat #监控 #Qwen3-14B # 大模型部署 # 私有化AI #gitea #screen 命令 #文心一言 #AI智能体 #vp9 #攻防演练 #Java web #红队 #iBMC #UltraISO #Apple AI #Apple 人工智能 #FoundationModel #Summarize #SwiftUI #LLM #大语言模型 #指针 #anaconda #虚拟环境 #GB28181 #SIP信令 #SpringBoot #视频监控 #WT-2026-0001 #QVD-2026-4572 #smartermail #多线程 # CUDA # GLM-TTS # 数据安全 #claudeCode #content7 #xshell #host key #跳槽 #工作 #sql注入 #TTS私有化 # IndexTTS # 音色克隆 #odoo #appche #ip #Ubuntu #arm开发 #Modbus-TCP # ARM服务器 # 大模型推理 #muduo #TcpServer #accept #高并发服务器 #Miniconda #远程开发 #系统管理 #服务 #Emby #视频 #微PE # GLM # 服务连通性 #azure #版本控制 #Git入门 #开发工具 #代码托管 #openEuler #ceph #ambari #arm #ftp #sftp #YOLO识别 #YOLO环境搭建Windows #YOLO环境搭建Ubuntu #挖矿 #Linux病毒 #turn #LangFlow # 轻量化镜像 # 边缘计算 #网安应急响应 #OpenHarmony #远程连接 #数据恢复 #视频恢复 #视频修复 #RAID5恢复 #流媒体服务器恢复 #jupyter #aws #cpu #哈希算法 #WinSCP 下载安装教程 #SFTP #FTP工具 #服务器文件传输 #Ansible # CosyVoice3 # 批量部署 #媒体 # 高并发 #AI部署 # ms-swift #SSH跳转 #PN 结 #TTS #服务器线程 # SSL通信 # 动态结构体 #RWK35xx #语音流 #实时传输 #go #node # GPU集群 #Gateway #认证服务器集成详解 #超算中心 #PBS #lsf #服务器开启 TLS v1.2 #IISCrypto 使用教程 #TLS 协议配置 #IIS 安全设置 #服务器运维工具 #excel #报表制作 #职场 #数据可视化 #信息可视化 #用数据讲故事 #uniapp #合法域名校验出错 #服务器域名配置不生效 #request域名配置 #已经配置好了但还是报错 #uniapp微信小程序 #框架搭建 #uni-app #H5 #手机h5网页浏览器 #安卓app #苹果ios APP #手机电脑开启摄像头并排查 #语音生成 #状态模式 #AI-native #dba #Tokio #华为od #华为机试 #Java #AI写作 #能源 #汽车 #Socket #套接字 #I/O多路复用 #字节序 #html5 #weston #x11 #x11显示服务器 #研发管理 #禅道 #禅道云端部署 #计算几何 #斜率 #方向归一化 #叉积 #samba #RSO #机器人操作系统 #lvs #glibc #宝塔面板部署RustDesk #RustDesk远程控制手机 #手机远程控制 #铁路桥梁 #DIC技术 #箱梁试验 #裂纹监测 #四点弯曲 #后端框架 #可再生能源 #绿色算力 #风电 #麦克风权限 #访问麦克风并录制音频 #麦克风录制音频后在线播放 #用户拒绝访问麦克风权限怎么办 #uniapp 安卓 苹果ios #将音频保存本地或上传服务器 #zabbix #深度优先 #DFS #集成学习 #express #cherry studio #Node.js # child_process #https #fabric #KMS #slmgr #winscp #AI应用编程 #前端框架 # 自动化运维 #.net #JNI #CPU #pxe #若依 # 数字人系统 # 远程部署 #GLM-4.6V-Flash-WEB # AI视觉 # 本地部署 #毕设 #STUN # TURN # NAT穿透 #MCP服务器注解 #异步支持 #方法筛选 #声明式编程 #自动筛选机制 #自由表达演说平台 #演说 #程序员创富 #rustdesk #p2p #Docker #3d #free #vmstat #sar #前端开发 #EN4FE #r语言 #mybatis #Karalon #AI Test #YOLOv8 # Docker镜像 #流程图 #论文阅读 #论文笔记 #图论 #spine #TRO #TRO侵权 #TRO和解 #国产开源制品管理工具 #Hadess #一文上手 #运维工具 #蓝桥杯 #okhttp #网络攻击模型 #进程 #进程创建与终止 #范式 #harmonyos #Discord机器人 #云部署 #程序那些事 #移动端h5网页 #调用浏览器摄像头并拍照 #开启摄像头权限 #拍照后查看与上传服务器端 #摄像头黑屏打不开问题 #企业微信 #服务器IO模型 #非阻塞轮询模型 #多任务并发模型 #异步信号模型 #多路复用模型 #系统安全 #小艺 #鸿蒙 #搜索 #ipmitool #BMC # 黑屏模式 # TTS服务器 #ollama #llm #领域驱动 #健康医疗 #AI应用 #文件IO #输入输出流 #图像识别 #高考 #串口服务器 #工业级串口服务器 #串口转以太网 #串口设备联网通讯模块 #串口服务器选型 #embedding #IndexTTS2 # 阿里云安骑士 # 木马查杀 #工程实践 #入侵 #日志排查 #kmeans #聚类 #时序数据库 #gpt #API #taro #wps #Linux多线程 #人大金仓 #Kingbase #Beidou #北斗 #SSR #代理模式 #Spring AOP #程序人生 #国产操作系统 #麒麟 #V11 #kylinos #bash #信息安全 #信息收集 #软件 #本地生活 #电商系统 #商城 #多进程 #python技巧 #高级IO #poll #企业级存储 #网络设备 #word #iot #软件工程 #Playbook #AI服务器 #生信 #pdf #Smokeping #STDIO协议 #Streamable-HTTP #McpTool注解 #服务器能力 #simulink #matlab #策略模式 #pve #AutoDL #GPU #租显卡 #训练推理 #aiohttp #asyncio #异步 #智能体来了 #传统行业 #AI赋能 #java大文件上传 #java大文件秒传 #java大文件上传下载 #java文件传输解决方案 #bigtop #hdp #hue #kerberos #pencil #pencil.dev #设计 #zotero #WebDAV #同步失败 #轻量化 #低配服务器 #.netcore #Anything-LLM #IDC服务器 #私有化部署 #大模型应用 #API调用 #PyInstaller打包运行 #服务端部署 #raid #raid阵列 #LoRA # lora-scripts # 模型微调 #VMware创建虚拟机 #远程更新 #缓存更新 #多指令适配 #物料关联计划 #挖漏洞 #攻击溯源 #编程 #blender #warp #PyCharm # 远程调试 # YOLOFuse #欧拉 #journalctl # GLM-4.6V-Flash-WEB # AI部署 #材料工程 #数码相机 #智能电视 #Langchain-Chatchat # 国产化服务器 # 信创 #net core #kestrel #web-server #asp.net-core #m3u8 #HLS #移动端H5网页 #APP安卓苹果ios #监控画面 直播视频流 #Syslog #系统日志 #日志分析 #日志监控 #Prometheus #生产服务器问题查询 #日志过滤 #Autodl私有云 #深度服务器配置 #Zabbix #语音合成 # 水冷服务器 # 风冷服务器 #二值化 #Canny边缘检测 #轮廓检测 #透视变换 #DooTask #VoxCPM-1.5-TTS # 云端GPU # PyCharm宕机 #Clawdbot #儿童AI #图像生成 #Qwen #防毒面罩 #防尘面罩 #pjsip #openresty #lua #AI生成 # outputs目录 # 自动化 #stl #漏洞修复 #IIS Crypto #SSH保活 #everything #postman #人脸识别sdk #视频编解码 #人脸识别 #flutter #esp32 arduino #select #决策树 #HistoryServer #Spark #YARN #jobhistory #身体实验室 #健康认知重构 #系统思维 #微行动 #NEAT效应 #亚健康自救 #ICT人 #ZooKeeper #ZooKeeper面试题 #面试宝典 #深入解析 #ComfyUI # 推理服务器 #云服务器 #个人电脑 #KMS 激活 #n8n解惑 #编程助手 #MC #MC群组服务器 #elk #rabbitmq # 显卡驱动备份 #模拟退火算法 #CS2 #debian13 #计算机毕业设计 #程序定制 #毕设代做 #课设 #源码 #VMware #BoringSSL #简单数论 #埃氏筛法 #Hadoop #远程控制 #云计算运维 #asp.net大文件上传 #asp.net大文件上传下载 #asp.net大文件上传源码 #ASP.NET断点续传 #asp.net上传文件夹 #asp.net上传大文件 #漏洞挖掘 #内存接口 # 澜起科技 # 服务器主板 #SSH别名 #大模型入门 #ci/cd #k8s #homelab #Lattepanda #Jellyfin #Plex #Kodi #yolov12 #研究生life #开关电源 #热敏电阻 #PTC热敏电阻 #文件传输 #电脑文件传输 #电脑传输文件 #电脑怎么传输文件到另一台电脑 #电脑传输文件到另一台电脑 #eureka #ICE #信创国产化 #达梦数据库 #mongodb #群晖 #wireshark #广播 #组播 #并发服务器 #nacos #银河麒麟aarch64 # 鲲鹏 #uvicorn #uvloop #asgi #event #http头信息 # 服务器迁移 # 回滚方案 #银河麒麟服务器系统 #远程桌面 #企业存储 #RustFS #对象存储 #高可用 #notepad++ #es安装 #鸿蒙PC #gpu #nvcc #cuda #nvidia ##租显卡 #树莓派 #温湿度监控 #WhatsApp通知 #IoT #MySQL #TensorRT # 推理优化 #TCP服务器 #开发实战 #全文检索 #可撤销IBE #服务器辅助 #私钥更新 #安全性证明 #双线性Diffie-Hellman #log4j #Jetty # 嵌入式服务器 #Kylin-Server #服务器安装 #模块 #Android16 #音频性能实战 #音频进阶 #短剧 #短剧小程序 #短剧系统 #微剧 #RXT4090显卡 #RTX4090 #深度学习服务器 #硬件选型 #hibernate #nosql #音乐 #IntelliJ IDEA #Spring Boot #neo4j #NoSQL #SQL #Llama-Factory #Coturn #TURN #新人首发 # 代理转发 # 跳板机 #VMWare Tool #echarts # 服务器IP # 端口7860 #建筑缺陷 #红外 #数据集 #SMARC #ARM #IO #网络安全大赛 # 公钥认证 #Reactor # 智能运维 # 性能瓶颈分析 #空间计算 #原型模式 #VibeVoice # 云服务器 #无人机 #CNAS #CMA #程序文件 #junit #web服务器 #实时检测 #卷积神经网络 #I/O模型 #并发 #水平触发、边缘触发 #多路复用 #googlecloud #DAG #clickhouse #云服务器选购 #Saas #线程 #代理 #数据访问 #outlook #错误代码2603 #无网络连接 #2603 #HarmonyOS APP #DHCP #C++ UA Server #SDK #Windows #跨平台开发 #agent #ai大模型 #eclipse #servlet #arm64 #具身智能 #SSH密钥 #SSH复用 # 远程开发 #磁盘配额 #存储管理 #形考作业 #国家开放大学 #系统运维 #自动化运维 #ETL管道 #RAG #向量存储 #数据预处理 #DocumentReader #GATT服务器 #蓝牙低功耗 #idea #intellij idea #lucene #散列表 #机器视觉 #6D位姿 #UOS #海光K100 #统信 #AI电商客服 #mssql #spring ai #oauth2 #wpf #rtmp #Modbus #MOXA #密码 #firefox #safari # 高温监控 # RTX 3090 #fs7TF #ROS # ControlMaster # 局域网访问 # 批量处理 #ui #cosmic #硬件 #Fun-ASR # 语音识别 # WebUI # 树莓派 # ARM架构 #跨域 #发布上线后跨域报错 #请求接口跨域问题解决 #跨域请求代理配置 #request浏览器跨域 #windbg分析蓝屏教程 #le audio #低功耗音频 #通信 #连接 #npu #大剑师 #nodejs面试题 #nmodbus4类库使用教程 #docker-compose #目标跟踪 #PowerBI #企业 #处理器 #IFix #ESP32 # OTA升级 # 黄山派 #内网 #c++20 # 远程连接 # 网络延迟 #Buck #NVIDIA #算力 #交错并联 #DGX #远程软件 #内存治理 #游戏机 #银河麒麟操作系统 #openssh #华为交换机 #信创终端 #UDP的API使用 #代理服务器 #rsync # 数据同步 #gerrit #opc ua #opc #设计师 #图像处理 #游戏美术 #技术美术 # 环境迁移 # Connection refused #matplotlib #teamviewer #安全架构 #飞牛nas #fnos #chat #HarmonyOS #雨云服务器 #Minecraft服务器 #教程 #MCSM面板 #Socket网络编程 # 串口服务器 # NPort5630 # 服务器配置 # GPU #Python办公自动化 #Python办公 #工程设计 #预混 #扩散 #燃烧知识 #层流 #湍流 #量子计算 #copilot #个人博客 #硬盘克隆 #DiskGenius # 键鼠锁定 #opc模拟服务器 #反向代理 #政务 #powerbi #个人助理 #数字员工 #参数估计 #矩估计 #概率论 #adobe #mtgsig #美团医药 #美团医药mtgsig #美团医药mtgsig1.2 #系统安装 #gmssh #宝塔 #1panel #Exchange #scikit-learn #随机森林 #闲置物品交易系统 #静脉曲张 #腿部健康 #运动 #IPv6 #POC #问答 #交付 #AI Agent #开发者工具 #MinIO #Minecraft #PaperMC #我的世界服务器 #sentinel #边缘AI # Kontron # SMARC-sAMX8 #jetty #kong #Kong Audio #Kong Audio3 #KongAudio3 #空音3 #空音 #中国民乐 #计算机外设 #ET模式 #非阻塞 #凤希AI伴侣 #remote-ssh #SA-PEKS # 关键词猜测攻击 # 盲签名 # 限速机制 #scanf #printf #getchar #putchar #cin #cout #CMake #Make #C/C++ #OpenAI #故障 #优化 #多模态 #微调 #超参 #LLamafactory #产品经理 #就业 #vps #AI论文写作工具 #学术写作辅助 #论文创作效率提升 #AI写论文实测 #docker安装seata #数字化转型 #实体经济 #商业模式 #软件开发 #数智红包 #商业变革 #创业干货 #AB包 # IndexTTS 2.0 #FASTMCP #全链路优化 #实战教程 #database #Go并发 #高并发架构 #Goroutine #系统设计 #Tracker 服务器 #响应最快 #torrent 下载 #2026年 #Aria2 可用 #迅雷可用 #BT工具通用 #联机教程 #局域网联机 #局域网联机教程 #局域网游戏 #交换机 #三层交换机 #高斯溅射 #UEFI #BIOS #Legacy BIOS #eBPF #sglang #云开发 #Harbor #AI智能棋盘 #Rock Pi S #边缘计算 #SSH Agent Forwarding # 容器化 #PTP_1588 #gPTP #c++高并发 #百万并发 #性能 #RAM # 权限修复 #uip # HiChatBox # 离线AI #X11转发 #SMTP # 内容安全 # Qwen3Guard #改行学it #平板 #零售 #智能硬件 #vncdotool #链接VNC服务器 #如何隐藏光标 #CTF #插件 #开源软件 #FHSS #Deepoc #具身模型 #开发板 #未来 #MinIO服务器启动与配置详解 #服务器解析漏洞 #NFC #智能公交 #服务器计费 #FP-增长 #算力建设 #gateway #Comate #Proxmox VE #虚拟化 #声源定位 #MUSIC #网路编程 #smtp #smtp服务器 #PHP #b树 # 远程访问 #Streamlit #AI聊天机器人 #tensorflow #memcache #memory mcp #Cursor #ansys #ansys问题解决办法 #分布式数据库 #集中式数据库 #业务需求 #选型误 #智能体对传统行业冲击 #行业转型