• TingWebServer服务器代码解读02

TingWebServer服务器代码解读02

2025-05-07 09:00:07 0 阅读

上一篇:TingWebServer服务器代码解读01

TingWebServer服务器代码解读01-CSDN博客

我们将跟随这个头文件包含图,继续逐级解读代码,我们解读的顺序还是从上往下的,这样能清楚的了解tingwebserver的服务器的框架包含结构,但在解读上一级文件的时候往往会运用到下一级的包含代码,因此希望读者在翻阅代码解读的时候多开几个窗口,以便在读到相关子集代码的时候能迅速找到对于代码解析。

这篇将进入TingWebserver的文件夹中解读相关文件,从thereadpool和http_conn.h解读到最后的日志文件。

Threadpool文件夹

README:

半同步/半反应堆线程池

使用一个工作队列完全解除了主线程和工作线程的耦合关系:主线程往工作队列中插入任务,工作线程通过竞争来取得任务并执行它。

  • 同步I/O模拟proactor模式
  • 半同步/半反应堆
  • 线程池

threadpool.h

跳过头文件

线程池定义:

template 
class threadpool
{
public:
    //构造函数,传参:工作模式,数据库连接池,线程数,最大请求量
    threadpool(int actor_model, connection_pool *connPool, int thread_number = 8, int max_request = 10000);
    ~threadpool();
    //把请求添加到工作队列当中
    bool append(T *request, int state);
    bool append_p(T *request);

private:
    static void *worker(void *arg);
    void run();

private:
    int m_thread_number;        // 线程池中的线程数
    int m_max_requests;         // 请求队列中允许的最大请求数
    pthread_t *m_threads;       // 线程池的线程数组
    std::list m_workqueue; // 请求队列,存储待处理的请求
    locker m_queuelocker;       // 保护请求队列的互斥锁
    sem m_queuestat;            // 信号量,表示请求队列中的任务数量
    connection_pool *m_connPool;  // 数据库连接池
    int m_actor_model;          // 工作模型,支持不同的请求处理模式
};

构造函数:threadpool

  • 初始化参数:构造函数接收参数来设置线程池的大小、请求队列的最大容量、工作模型以及数据库连接池。
  • 线程池创建:首先检查线程数和最大请求数是否合法,如果不合法则抛出异常。然后为线程池分配内存并创建线程。每个线程会运行 worker 函数,并传递 this 指针作为参数。
  • 线程分离:使用 pthread_detach 来分离线程,这样每个线程结束后会自动释放资源,而不需要手动 join
  • std::list m_workqueue 存储的是任务对象的指针,这样可以避免对象的拷贝,并且能够更好地管理对象的生命周期
template 
threadpool::threadpool(int actor_model, connection_pool *connPool, int thread_number, int max_requests)
    : m_actor_model(actor_model), m_thread_number(thread_number), m_max_requests(max_requests), m_threads(NULL), m_connPool(connPool)
//类内成员初始化
{
    if (thread_number <= 0 || max_requests <= 0)
        throw std::exception();  // 如果线程数或最大请求数不合法,抛出异常
    m_threads = new pthread_t[m_thread_number];  // 创建线程池
    if (!m_threads)
        throw std::exception();
    
    // 创建工作线程
    for (int i = 0; i < thread_number; ++i)
    {
        if (pthread_create(m_threads + i, NULL, worker, this) != 0)
        {
            delete[] m_threads;
            throw std::exception();  // 创建线程失败,抛出异常
        }
        if (pthread_detach(m_threads[i]))  // 分离线程
        {
            delete[] m_threads;
            throw std::exception();  // 分离线程失败,抛出异常
        }
    }
}

 析构函数 :~threadpool

template 
threadpool::~threadpool()
{
    delete[] m_threads;  // 释放线程池的内存
}

任务队列操作:appendappend_p 

template 
bool threadpool::append(T *request, int state)
{
    m_queuelocker.lock();//先加锁,再操作
    if (m_workqueue.size() >= m_max_requests)  // 如果请求队列已满,返回 false
    {
        m_queuelocker.unlock();
        return false;
    }
    request->m_state = state;  // 设置请求的状态
    m_workqueue.push_back(request);  // 将请求添加到队列
    m_queuelocker.unlock();//结束操作,解锁
    m_queuestat.post();  // 通知有新的请求
    return true;
}

append_p是一个不用state参数的加入请求队列函数 

template 
bool threadpool::append_p(T *request)
{
    m_queuelocker.lock();
    if (m_workqueue.size() >= m_max_requests)  // 如果请求队列已满,返回 false
    {
        m_queuelocker.unlock();
        return false;
    }
    m_workqueue.push_back(request);  // 将请求添加到队列
    m_queuelocker.unlock();
    m_queuestat.post();  // 通知有新的请求
    return true;
}

工作线程:workerrun

在多线程编程中,线程的执行函数(即线程入口函数)必须是一个符合操作系统要求的格式。在 POSIX 线程(pthread)中,线程入口函数的原型是: 

void* thread_func(void* arg);

直接使用run函数,因为run函数没有传入参数void*,所以创建一个中介函数worker,将传递给线程的参数从void*转化为threadpool*类型,从而可以调用run函数,上面的pthread_create传入了this(本线程),当作worker的传入参数void*arg,从而调用pool的run函数

worker函数: 
template 
void *threadpool::worker(void *arg)
{
    threadpool *pool = (threadpool *)arg;
    pool->run();  // 每个线程调用 run 函数
    return pool;
}
run函数:

进入循环,然后信号m_queuestat进行阻塞直到有新任务,上锁,判断队列是否为空,是则解锁继续等待,否则进行处理,获取第一个请求,(在wait到了信号后,进行 m_queuestat内部的信号处理,m_wirkqueue队列+1),读取请求后移除,进行解锁,然后根据不同的操作模式进行不同的操作

Reactor 模式:当 m_actor_model 为 1 时,表示线程池采用的是 Reactor 模式。

  • 读取请求:如果请求的状态 request->m_state 为 0,表示该请求是读取操作。此时,调用 request->read_once() 读取数据:

    • 如果读取成功,则设置 request->improv = 1,表示请求已被改进,接着创建一个 connectionRAII 对象来管理数据库连接,最后调用 request->process() 处理请求。
    • 如果读取失败,设置 request->timer_flag = 1,表示该请求读取失败,需要设置定时器进行超时处理。
  • 写入请求:如果请求的状态是写入操作(request->m_state 为 1),调用 request->write() 写入数据:

    • 如果写入成功,设置 request->improv = 1
    • 如果写入失败,设置 request->timer_flag = 1,表示写入失败,需要进行超时处理

Proactor 模式:当 m_actor_model 不为 1 时,表示线程池采用的是 Proactor 模式。

  • 在 Proactor 模式下,线程池会直接从请求队列中取出请求并进行处理。connectionRAII 是用于管理数据库连接的智能指针,确保数据库连接在请求处理完后自动释放。

  • 然后调用 request->process() 处理请求。

template 
void threadpool::run()
{
    while (true)//进行循环
    {
        m_queuestat.wait();  // 等待任务
        m_queuelocker.lock();//上锁
        if (m_workqueue.empty())  // 如果请求队列为空,则继续等待
        {
            m_queuelocker.unlock();
            continue;
        }
        T *request = m_workqueue.front();  // 获取队列中的第一个请求
        m_workqueue.pop_front();  // 从队列中移除请求
        m_queuelocker.unlock();
        
        //如果请求为空,跳过当前循环进入下一次,防止程序崩溃
        if (!request)
            continue;
        
        // 根据 actor_model 选择不同的处理方式
        if (1 == m_actor_model)  // Reactor 模式
        {
            if (0 == request->m_state)  // 读取请求
            {
                if (request->read_once())  // 读取成功
                {
                    request->improv = 1;  // 设置请求改为改进状态
                    connectionRAII mysqlcon(&request->mysql, m_connPool);  // 获取数据库连接
                    request->process();  // 处理请求
                }
                else
                {
                    request->improv = 1;
                    request->timer_flag = 1;  // 请求读取失败,设置超时标志
                }
            }
            else  // 写入请求
            {
                if (request->write())  // 写入成功
                {
                    request->improv = 1;
                }
                else
                {
                    request->improv = 1;
                    request->timer_flag = 1;  // 请求写入失败,设置超时标志
                }
            }
        }
        else  // Proactor 模式
        {
            connectionRAII mysqlcon(&request->mysql, m_connPool);
            request->process();  // 处理请求
        }
    }
}

Http文件夹

README:

http连接处理类

根据状态转移,通过主从状态机封装了http连接类。其中,主状态机在内部调用从状态机,从状态机将处理状态和数据传给主状态机

  • 客户端发出http连接请求
  • 从状态机读取数据,更新自身状态和接收数据,传给主状态机
  • 主状态机根据从状态机状态,更新自身状态,决定响应请求还是继续读取

http_conn.h

#ifndef HTTPCONNECTION_H
#define HTTPCONNECTION_H

// 包含头文件,提供系统调用、网络功能和其他操作所需的库
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#include "../lock/locker.h"  // 锁的操作
#include "../CGImysql/sql_connection_pool.h"  // MySQL连接池
#include "../timer/lst_timer.h"  // 定时器
#include "../log/log.h"  // 日志管理

class http_conn
{
public:
    // 定义常量
    static const int FILENAME_LEN = 200;  // 文件名最大长度
    static const int READ_BUFFER_SIZE = 2048;  // 读缓冲区大小
    static const int WRITE_BUFFER_SIZE = 1024;  // 写缓冲区大小

    // HTTP请求的方法
    enum METHOD
    {
        GET = 0,
        POST,
        HEAD,
        PUT,
        DELETE,
        TRACE,
        OPTIONS,
        CONNECT,
        PATH
    };

    // 检查请求的不同状态
    enum CHECK_STATE
    {
        CHECK_STATE_REQUESTLINE = 0,  // 请求行检查状态
        CHECK_STATE_HEADER,  // 请求头检查状态
        CHECK_STATE_CONTENT  // 请求内容检查状态
    };

    // HTTP状态码
    enum HTTP_CODE
    {
        NO_REQUEST,        // 没有请求
        GET_REQUEST,       // GET 请求
        BAD_REQUEST,       // 错误请求
        NO_RESOURCE,       // 无资源
        FORBIDDEN_REQUEST, // 禁止请求
        FILE_REQUEST,      // 文件请求
        INTERNAL_ERROR,    // 内部错误
        CLOSED_CONNECTION  // 关闭连接
    };

    // 行的解析状态
    enum LINE_STATUS
    {
        LINE_OK = 0,  // 行格式正确
        LINE_BAD,     // 行格式错误
        LINE_OPEN     // 行未完全接收
    };

public:
    http_conn() {}  // 构造函数
    ~http_conn() {}  // 析构函数

public:
    // 初始化 HTTP 连接
    void init(int sockfd, const sockaddr_in &addr, char *root, int trigmode, int close_log, string user, string passwd, string sqlname);
    // 关闭连接
    void close_conn(bool real_close = true);
    // 处理请求
    void process();
    // 读取请求数据
    bool read_once();
    // 写响应数据
    bool write();
    // 获取客户端的地址
    sockaddr_in *get_address()
    {
        return &m_address;
    }
    // 初始化数据库连接
    void initmysql_result(connection_pool *connPool);

    // 定时器标志和改进标志
    int timer_flag;
    int improv;

private:
    // 内部初始化函数
    void init();
    // 处理读取请求的函数
    HTTP_CODE process_read();
    // 处理写入响应的函数
    bool process_write(HTTP_CODE ret);
    // 解析请求行
    HTTP_CODE parse_request_line(char *text);
    // 解析请求头
    HTTP_CODE parse_headers(char *text);
    // 解析请求内容
    HTTP_CODE parse_content(char *text);
    // 执行请求
    HTTP_CODE do_request();
    // 获取当前行的指针
    char *get_line() { return m_read_buf + m_start_line; };
    // 解析一行数据
    LINE_STATUS parse_line();
    // 解除内存映射
    void unmap();
    // 添加响应头
    bool add_response(const char *format, ...);
    // 添加响应内容
    bool add_content(const char *content);
    // 添加状态行
    bool add_status_line(int status, const char *title);
    // 添加响应头
    bool add_headers(int content_length);
    // 添加内容类型
    bool add_content_type();
    // 添加内容长度
    bool add_content_length(int content_length);
    // 添加连接状态
    bool add_linger();
    // 添加空行
    bool add_blank_line();

public:
    // 静态成员变量,所有 http_conn 对象共享
    static int m_epollfd;  // epoll 文件描述符,用于事件通知
    static int m_user_count;  // 当前连接的用户数

    MYSQL *mysql;  // MySQL 数据库连接
    int m_state;  // 请求的状态(读为 0,写为 1)

private:
    int m_sockfd;  // 客户端 socket 文件描述符
    sockaddr_in m_address;  // 客户端地址信息
    char m_read_buf[READ_BUFFER_SIZE];  // 读缓冲区
    long m_read_idx;  // 当前读取到缓冲区的字节数
    long m_checked_idx;  // 已检查的字节数
    int m_start_line;  // 当前行的起始位置
    char m_write_buf[WRITE_BUFFER_SIZE];  // 写缓冲区
    int m_write_idx;  // 当前写入的字节数
    CHECK_STATE m_check_state;  // 当前请求的检查状态
    METHOD m_method;  // 请求方法(GET、POST等)
    char m_real_file[FILENAME_LEN];  // 请求的文件路径
    char *m_url;  // 请求的URL
    char *m_version;  // HTTP版本
    char *m_host;  // Host头
    long m_content_length;  // 内容长度
    bool m_linger;  // 是否保持连接
    char *m_file_address;  // 文件内存地址
    struct stat m_file_stat;  // 文件状态
    struct iovec m_iv[2];  // 用于writev的写缓冲区
    int m_iv_count;  // 写缓冲区数量
    int cgi;  // 是否启用POST方法
    char *m_string;  // 存储请求头数据
    int bytes_to_send;  // 要发送的字节数
    int bytes_have_send;  // 已发送的字节数
    char *doc_root;  // 网站根目录

    // 存储用户信息的map
    map m_users;

    // 配置相关
    int m_TRIGMode;  // 触发模式
    int m_close_log;  // 是否关闭日志

    // 数据库连接的配置信息
    char sql_user[100];
    char sql_passwd[100];
    char sql_name[100];
};

#endif
  • 常量定义

    • FILENAME_LEN:文件名最大长度。
    • READ_BUFFER_SIZE:读取缓冲区大小。
    • WRITE_BUFFER_SIZE:写入缓冲区大小。
  • 枚举类型

    • METHOD:HTTP请求方法类型(如 GET、POST、DELETE 等)。
    • CHECK_STATE:HTTP请求的检查状态(请求行、请求头、请求内容)。
    • HTTP_CODE:HTTP响应代码(如 NO_REQUEST, GET_REQUEST, BAD_REQUEST 等)。
    • LINE_STATUS:解析请求行时的状态。
  • 函数

    • init():初始化HTTP连接,绑定文件描述符和客户端地址等。
    • close_conn():关闭连接,释放相关资源。
    • process():处理请求,负责请求的读取、解析、响应等。
    • read_once():从客户端读取请求数据。
    • write():向客户端写入响应数据。
  • MySQL相关

    • initmysql_result():初始化MySQL连接。
    • m_users:用于存储用户信息。
  • 文件相关

    • m_real_file:存储文件的路径名。
    • m_file_stat:存储文件的状态信息。
  • 定时器和触发模式

    • timer_flag:定时器标志,用于判断是否需要关闭连接。
    • m_TRIGMode:触发模式(如边缘触发或水平触发)。

http_conn.cpp

头文件以及宏定义


#include "http_conn.h"

#include 
#include 

// 定义 HTTP 响应的一些状态信息
const char *ok_200_title = "OK"; // 状态码 200 的响应标题
const char *error_400_title = "Bad Request"; // 状态码 400 的响应标题
const char *error_400_form = "Your request has bad syntax or is inherently impossible to satisfy.
"; // 状态码 400 错误信息
const char *error_403_title = "Forbidden"; // 状态码 403 的响应标题
const char *error_403_form = "You do not have permission to get file from this server.
"; // 状态码 403 错误信息
const char *error_404_title = "Not Found"; // 状态码 404 的响应标题
const char *error_404_form = "The requested file was not found on this server.
"; // 状态码 404 错误信息
const char *error_500_title = "Internal Error"; // 状态码 500 的响应标题
const char *error_500_form = "There was an unusual problem serving the request file.
"; // 状态码 500 错误信息

获取数据库信息函数:initmysql_result

把数据库中保存的user和password映射倒map中,map的结构为:

map users;

string 类型的 key 表示用户名,value 表示密码。这样就可以通过用户名在 users map 中查找对应的密码,从而实现简单的用户验证机制

connectionRAII mysqlcon(&mysql, connPool) 是一个栈对象,它的构造函数会从数据库连接池 connPool 中获取一个 MySQL 连接,并将其赋值给 mysqlconnectionRAII 可能是一个自定义的 RAII(资源获取即初始化)类,确保在作用域结束时自动释放数据库连接。

mysql_query 是 MySQL C API 中的函数,用于执行 SQL 查询。它执行查询 "SELECT username,passwd FROM user",即从 user 表中选择 usernamepasswd 字段 

mysql_store_result(mysql) 获取查询结果并返回 MYSQL_RES 类型的指针 result。这个指针指向存储了 SQL 查询结果的结构体。 

mysql_num_fields(result) 返回结果集中的字段(列)数,即 user 表中的列数。 

mysql_fetch_fields(result) 返回一个 MYSQL_FIELD 数组,包含每个字段的元数据(例如字段名、数据类型等)。 

  • mysql_fetch_row(result) 从结果集中获取一行数据,返回一个 MYSQL_ROW 类型的指针,row 是一个包含字段值的数组。
    • row[0] 是当前行的第一个字段,即 username
    • row[1] 是当前行的第二个字段,即 passwd
  • 通过 string 构造函数将 usernamepasswd 转换为 string 类型的 temp1temp2
  • 使用 users[temp1] = temp2;username 作为 keypasswd 作为 value 存入全局的 users map。

void http_conn::initmysql_result(connection_pool *connPool)
{
    //先从连接池中取一个连接
    MYSQL *mysql = NULL;
    connectionRAII mysqlcon(&mysql, connPool);

    //在user表中检索username,passwd数据,浏览器端输入
    if (mysql_query(mysql, "SELECT username,passwd FROM user"))
    {
        LOG_ERROR("SELECT error:%s
", mysql_error(mysql));
    }

    //从表中检索完整的结果集
    MYSQL_RES *result = mysql_store_result(mysql);

    //返回结果集中的列数
    int num_fields = mysql_num_fields(result);

    //返回所有字段结构的数组
    MYSQL_FIELD *fields = mysql_fetch_fields(result);

    //从结果集中获取下一行,将对应的用户名和密码,存入map中
    while (MYSQL_ROW row = mysql_fetch_row(result))
    {
        string temp1(row[0]);
        string temp2(row[1]);
        users[temp1] = temp2;
    }
}

文件描述符设置非阻塞函数:setnonblocking

设置非阻塞

//对文件描述符设置非阻塞
int setnonblocking(int fd)
{
    int old_option = fcntl(fd, F_GETFL);
    int new_option = old_option | O_NONBLOCK;
    fcntl(fd, F_SETFL, new_option);
    return old_option;
}

注册读事件函数:addfd

//将内核事件表注册读事件,ET模式,选择开启EPOLLONESHOT
void addfd(int epollfd, int fd, bool one_shot, int TRIGMode)
{
    epoll_event event;
    event.data.fd = fd;

    if (1 == TRIGMode)//根据trig模式设置为ET或者LT触发模式
        event.events = EPOLLIN | EPOLLET | EPOLLRDHUP;//ET
    else
        event.events = EPOLLIN | EPOLLRDHUP;//LT

    if (one_shot)//选择开启oneshot
        event.events |= EPOLLONESHOT;
    epoll_ctl(epollfd, EPOLL_CTL_ADD, fd, &event);//挂上红黑树
    setnonblocking(fd);//设置非阻塞
}

删除文件描述符:removefd

//从内核时间表删除描述符
void removefd(int epollfd, int fd)
{
    epoll_ctl(epollfd, EPOLL_CTL_DEL, fd, 0);
    close(fd);
}

事件重置为EPOLLONESHOT函数:modfd

不同触发模式不同event

void modfd(int epollfd, int fd, int ev, int TRIGMode)
{
    epoll_event event;
    event.data.fd = fd;

    if (1 == TRIGMode)
        event.events = ev | EPOLLET | EPOLLONESHOT | EPOLLRDHUP;
    else
        event.events = ev | EPOLLONESHOT | EPOLLRDHUP;

    epoll_ctl(epollfd, EPOLL_CTL_MOD, fd, &event);
}

类外初始化静态变量:

int http_conn::m_user_count = 0;
int http_conn::m_epollfd = -1;

m_user_count 用于记录当前活动的 http_conn 实例的数量:连接的用户数。

m_epollfd 是与 epoll 相关的文件描述符,通常用于处理 I/O 多路复用。-1 是一个常用的表示未初始化或出错的标志。

关闭连接函数:close_conn

void http_conn::close_conn(bool real_close)//real_close表示是否需要执行关闭操作
{
    if (real_close && (m_sockfd != -1))//m_sockfd != -1表示当前套接字是否有效
    {
        printf("close %d
", m_sockfd);
        removefd(m_epollfd, m_sockfd);//移除文件描述符
        m_sockfd = -1;//表示连接关闭
        m_user_count--;//连接数量-1
    }
}

初始化连接函数:init

初始化一个新的http连接对象 ,把套接字和地址传给成员变量,调用addfd将当前套接字加到epoll实例中去,增加活跃连接数,配置网站根目录、触发模式、日志标志、数据库连接信息等,调用 init 函数完成其他内部初始化工作。

void http_conn::init(int sockfd, const sockaddr_in &addr, char *root, int TRIGMode,
                     int close_log, string user, string passwd, string sqlname)
{
    m_sockfd = sockfd;
    m_address = addr;

    addfd(m_epollfd, sockfd, true, m_TRIGMode);//注册读事件
    m_user_count++;//连接数量+1

    //当浏览器出现连接重置时,可能是网站根目录出错或http响应格式出错或者访问的文件中内容完全为空
    doc_root = root;
    m_TRIGMode = TRIGMode;
    m_close_log = close_log;

    strcpy(sql_user, user.c_str());
    strcpy(sql_passwd, passwd.c_str());//passwd 赋值给成员变量 sql_passwd
    strcpy(sql_name, sqlname.c_str());//sqlname 赋值给成员变量 sql_name

    init();//下面的void init()进行剩下的初始化工作
}

初始化新接受的连接:void init

 作用于前面的init中

// HTTP连接的初始化函数,设置所有与HTTP请求相关的成员变量的初始状态
void http_conn::init()
{
    mysql = NULL;  // 将MySQL连接指针初始化为NULL,表示尚未连接到数据库
    bytes_to_send = 0;  // 设置待发送的字节数为0
    bytes_have_send = 0;  // 设置已发送的字节数为0
    m_check_state = CHECK_STATE_REQUESTLINE;  // 设置解析状态机的初始状态为请求行分析状态
    m_linger = false;  // 设置HTTP请求不使用长连接(默认为短连接)
    m_method = GET;  // 默认为GET方法
    m_url = 0;  // 设置URL为NULL,表示没有设置请求的URL
    m_version = 0;  // 设置HTTP版本为NULL,表示未指定HTTP版本
    m_content_length = 0;  // 设置内容长度为0
    m_host = 0;  // 设置Host字段为NULL
    m_start_line = 0;  // 请求行的起始位置为0
    m_checked_idx = 0;  // 当前分析的字符位置为0
    m_read_idx = 0;  // 当前读取的字节位置为0
    m_write_idx = 0;  // 当前写入的字节位置为0
    cgi = 0;  // 默认不启用CGI功能(如POST请求)
    m_state = 0;  // 请求的状态初始化为0
    timer_flag = 0;  // 定时器标志初始化为0,表示没有定时器标记
    improv = 0;  // 改进标志初始化为0,表示没有改进状态

    memset(m_read_buf, '', READ_BUFFER_SIZE);  // 清空读取缓存区,将其初始化为全零
    memset(m_write_buf, '', WRITE_BUFFER_SIZE);  // 清空写入缓存区,将其初始化为全零
    memset(m_real_file, '', FILENAME_LEN);  // 清空存储真实文件路径的字符数组
}

 分析行函数:parse_line

  • parse_line() 函数的作用是从 m_read_buf 缓冲区中逐字符检查一行数据,并确保该行符合HTTP协议的格式(每行以 结尾)。
  • 如果符合格式,返回 LINE_OK;如果格式错误,返回 LINE_BAD;如果没有解析到完整的一行,返回 LINE_OPEN
  • 这个设计的目的是分步读取和检查HTTP请求的每一行,确保处理流程的严谨性。
char temp;
for (; m_checked_idx < m_read_idx; ++m_checked_idx)
{
    temp = m_read_buf[m_checked_idx];  // 获取当前要检查的字符
    if (temp == '
')  // 检查是否遇到回车符 '
'
    {
        if ((m_checked_idx + 1) == m_read_idx)  // 如果回车符后没有换行符,则当前行未结束
            return LINE_OPEN;  // 返回 LINE_OPEN,表示这一行没有结束,仍然在读取中
        else if (m_read_buf[m_checked_idx + 1] == '
')  // 如果回车符后紧跟着换行符 '
'
        {
            m_read_buf[m_checked_idx++] = '';  // 将回车符 '
' 置为 '',标记行结束
            m_read_buf[m_checked_idx++] = '';  // 将换行符 '
' 置为 '',标记行结束
            return LINE_OK;  // 返回 LINE_OK,表示成功解析出完整的一行
        }
        return LINE_BAD;  // 如果回车符后没有换行符,则返回 LINE_BAD,表示该行格式错误
    }
    else if (temp == '
')  // 检查是否遇到换行符 '
'
    {
        if (m_checked_idx > 1 && m_read_buf[m_checked_idx - 1] == '
')  // 如果换行符前有回车符 '
'
        {
            m_read_buf[m_checked_idx - 1] = '';  // 将回车符 '
' 置为 '',标记行结束
            m_read_buf[m_checked_idx++] = '';  // 将换行符 '
' 置为 '',标记行结束
            return LINE_OK;  // 返回 LINE_OK,表示成功解析出完整的一行
        }
        return LINE_BAD;  // 如果没有回车符前缀,则返回 LINE_BAD,表示该行格式错误
    }
}
return LINE_OPEN;  // 如果没有找到回车换行符,则返回 LINE_OPEN,表示该行还未结束

读取数据函数:read_once

  • LT 模式recv 每次读取一次,读取的数据量有限,每次调用都会返回已读取的字节数。如果没有数据可读取,recv 返回 0 或负数,函数根据返回值判断读取结果。
  • ET 模式recv 尽可能多地读取数据,并且只有在没有更多数据时才停止。此模式适用于高效读取,在一次 recv 调用后没有数据时不会再继续触发读取,直到有数据到来。
bool http_conn::read_once()
{
    if (m_read_idx >= READ_BUFFER_SIZE)//检查缓冲区是否已经填满
    {
        return false;
    }
    int bytes_read = 0;//记录本次读取字节数

    //LT读取数据
    if (0 == m_TRIGMode)
    {
        bytes_read = recv(m_sockfd, m_read_buf + m_read_idx, READ_BUFFER_SIZE - m_read_idx, 0);//m_read_buf + m_read_idx为本次数据存储下标
        m_read_idx += bytes_read;//更新下标

        if (bytes_read <= 0)//未读出数据
        {
            return false;
        }

        return true;
    }
    //ET读数据
    else
    {
        while (true)//循环读取数据,直至没有
        {
            bytes_read = recv(m_sockfd, m_read_buf + m_read_idx, READ_BUFFER_SIZE - m_read_idx, 0);
            if (bytes_read == -1)
            {
                if (errno == EAGAIN || errno == EWOULDBLOCK)//暂时没有可读取的数据,跳出进行下一次循环
                    break;
                return false;//其他错误则返回读取失败
            }
            else if (bytes_read == 0)//连接关闭
            {
                return false;
            }
            m_read_idx += bytes_read;//更新下标
        }
        return true;
    }
} 

解析http请求行函数:parse_request_line

解析传入http请求头,假设传入的是 GET /index.html HTTP/1.1 ,函数是从其中取出 HTTP 方法(GET)、URL(/index.html)和版本(HTTP/1.1

http_conn::HTTP_CODE http_conn::parse_request_line(char *text)
{
    m_url = strpbrk(text, " 	");//返回第一个空格的位置
    if (!m_url)
    {
        return BAD_REQUEST;
    }
    *m_url++ = '';//将空格转化为字符串终止符'/0'
    char *method = text;//mothod指向请求行开始
    if (strcasecmp(method, "GET") == 0)//匹配请求字符
        m_method = GET;
    else if (strcasecmp(method, "POST") == 0)
    {
        m_method = POST;
        cgi = 1;
    }
    else
        return BAD_REQUEST;//说明请求的方法不支持
    m_url += strspn(m_url, " 	");//跳过前面的字符,开始http请求处理
    m_version = strpbrk(m_url, " 	");//查找第一个空格位置
    if (!m_version)
        return BAD_REQUEST;
    *m_version++ = '';
    m_version += strspn(m_version, " 	");
    if (strcasecmp(m_version, "HTTP/1.1") != 0)
        return BAD_REQUEST;
    if (strncasecmp(m_url, "http://", 7) == 0)
    {
        m_url += 7;
        m_url = strchr(m_url, '/');
    }

    if (strncasecmp(m_url, "https://", 8) == 0)
    {
        m_url += 8;
        m_url = strchr(m_url, '/');
    }

    if (!m_url || m_url[0] != '/')
        return BAD_REQUEST;
    //当url为/时,显示判断界面
    if (strlen(m_url) == 1)
        strcat(m_url, "judge.html");
    m_check_state = CHECK_STATE_HEADER;
    return NO_REQUEST;
}

解析http请求的头部信息函数:parse_headers:

传入的text是上面parse_line函数切割出来的每一行的内容

该函数接收一个指向 HTTP 头部字段的字符串 text,然后解析:

  • Connection 头部(是否是 keep-alive
  • Content-Length 头部(请求体长度)
  • Host 头部(主机名)

如果 text 为空行,则意味着请求头解析结束:

  • 如果 Content-Length 不是 0,则说明请求有请求体,状态转换为 CHECK_STATE_CONTENT
  • 否则,解析完成,返回 GET_REQUEST
http_conn::HTTP_CODE http_conn::parse_headers(char *text)
{
    if (text[0] == '')//读取到空行
    {
        if (m_content_length != 0)//请求带有请求体(post请求)
        {
            m_check_state = CHECK_STATE_CONTENT;
            return NO_REQUEST;
        }
        return GET_REQUEST;//get请求,直接返回表示请求完成
    }
    else if (strncasecmp(text, "Connection:", 11) == 0)
    {
        text += 11;
        text += strspn(text, " 	");
        if (strcasecmp(text, "keep-alive") == 0)//表示HTTP是持久连接
        {
            m_linger = true;//表示连接不会立即关闭
        }
    }
    else if (strncasecmp(text, "Content-length:", 15) == 0)//指定请求体长度,通常用于POST
    {
        text += 15;
        text += strspn(text, " 	");
        m_content_length = atol(text);//字符串转化为long类型
    }
    else if (strncasecmp(text, "Host:", 5) == 0)//读入的是host
    {
        text += 5;
        text += strspn(text, " 	");
        m_host = text;//主机名保留到变量中
    }
    else
    {
        LOG_INFO("oop!unknow header: %s", text);//其他信息
    }
    return NO_REQUEST;
}

判断http请求是否被完整读入函数:parse_conte

当读取长度大于等于请求体长度+http头部长度,判断为请求体完整,进行处理

http_conn::HTTP_CODE http_conn::parse_content(char *text)
{
    if (m_read_idx >= (m_content_length + m_checked_idx))
    {
        text[m_content_length] = '';//手动添加字符串终止符
        //POST请求中最后为输入的用户名和密码
        m_string = text;//把请求体的数据存入 m_string 变量
        return GET_REQUEST;
    }
    return NO_REQUEST;
}

解析http主体函数:process_read

  • process_read() 是 HTTP 请求解析的核心,解析请求行、头部和请求体。
  • 使用 while 循环不断解析,确保完整接收 HTTP 请求。
  • 调用 parse_request_line()parse_headers()parse_content() 进行解析
  • 请求完整时调用 do_request() 处理请求,否则等待更多数据。
步骤m_check_statetext 解析内容ret 解析结果下一步
1CHECK_STATE_REQUESTLINEPOST /login HTTP/1.1NO_REQUEST进入 CHECK_STATE_HEADER
2CHECK_STATE_HEADERHost: www.example.comNO_REQUEST继续解析头部
3CHECK_STATE_HEADERContent-Length: 15NO_REQUEST继续解析
4CHECK_STATE_HEADER(空行,表示头部结束)NO_REQUEST进入 CHECK_STATE_CONTENT
5CHECK_STATE_CONTENTuser=admin&pwd=123GET_REQUEST调用 do_request() 处理
http_conn::HTTP_CODE http_conn::process_read()
{
    LINE_STATUS line_status = LINE_OK; // 记录当前行解析状态
    HTTP_CODE ret = NO_REQUEST;        // 记录 HTTP 请求的解析状态
    char *text = 0;                    // 存储当前解析的行文本

    // 解析 HTTP 请求的所有行
    while ((m_check_state == CHECK_STATE_CONTENT && line_status == LINE_OK) || ((line_status = parse_line()) == LINE_OK))
    {
        text = get_line();        // 获取当前行的起始地址
        m_start_line = m_checked_idx;
        LOG_INFO("%s", text);     // 记录日志,输出当前解析的文本行

        switch (m_check_state)
        {
        case CHECK_STATE_REQUESTLINE:
        {
            ret = parse_request_line(text); // 解析请求行
            if (ret == BAD_REQUEST)
                return BAD_REQUEST;
            break;
        }
        case CHECK_STATE_HEADER:
        {
            ret = parse_headers(text); // 解析请求头
            if (ret == BAD_REQUEST)
                return BAD_REQUEST;
            else if (ret == GET_REQUEST)
            {
                return do_request(); // 解析完成,调用 do_request() 处理请求
            }
            break;
        }
        case CHECK_STATE_CONTENT:
        {
            ret = parse_content(text); // 解析请求体
            if (ret == GET_REQUEST)
                return do_request();
            line_status = LINE_OPEN; // 请求体可能不止一行,继续解析
            break;
        }
        default:
            return INTERNAL_ERROR; // 状态异常,返回服务器内部错误
        }
    }
    return NO_REQUEST; // 如果解析未完成,则继续等待数据
}

处理静态文件请求和动态请求函数:do_request 

  • 动态请求处理: 如果是 登录注册 请求,会提取 POST 数据,执行数据库操作,返回不同的页面。
  • 静态资源处理: 解析 URL 和路径,根据不同的条件返回不同的静态页面。
  • 文件操作: 在文件请求时,检查文件存在、权限以及是否为目录。符合条件则映射文件至内存,准备响应。
http_conn::HTTP_CODE http_conn::do_request()
{
    strcpy(m_real_file, doc_root);
    int len = strlen(doc_root);//设置文件实质路径
    //printf("m_url:%s
", m_url);
    const char *p = strrchr(m_url, '/');

    //处理cgi
    if (cgi == 1 && (*(p + 1) == '2' || *(p + 1) == '3'))
    {

        //根据标志判断是登录检测还是注册检测
        char flag = m_url[1];

        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/");
        strcat(m_url_real, m_url + 2);
        strncpy(m_real_file + len, m_url_real, FILENAME_LEN - len - 1);
        free(m_url_real);

        //将用户名和密码提取出来
        //user=123&passwd=123
        char name[100], password[100];
        int i;
        for (i = 5; m_string[i] != '&'; ++i)
            name[i - 5] = m_string[i];
        name[i - 5] = '';

        int j = 0;
        for (i = i + 10; m_string[i] != ''; ++i, ++j)
            password[j] = m_string[i];
        password[j] = '';

        if (*(p + 1) == '3')
        {
            //如果是注册,先检测数据库中是否有重名的
            //没有重名的,进行增加数据
            char *sql_insert = (char *)malloc(sizeof(char) * 200);
            strcpy(sql_insert, "INSERT INTO user(username, passwd) VALUES(");
            strcat(sql_insert, "'");
            strcat(sql_insert, name);
            strcat(sql_insert, "', '");
            strcat(sql_insert, password);
            strcat(sql_insert, "')");

            if (users.find(name) == users.end())
            {
                m_lock.lock();
                int res = mysql_query(mysql, sql_insert);
                users.insert(pair(name, password));
                m_lock.unlock();

                if (!res)
                    strcpy(m_url, "/log.html");
                else
                    strcpy(m_url, "/registerError.html");
            }
            else
                strcpy(m_url, "/registerError.html");
        }
        //如果是登录,直接判断
        //若浏览器端输入的用户名和密码在表中可以查找到,返回1,否则返回0
        else if (*(p + 1) == '2')
        {
            if (users.find(name) != users.end() && users[name] == password)
                strcpy(m_url, "/welcome.html");
            else
                strcpy(m_url, "/logError.html");
        }
    }

    if (*(p + 1) == '0')
    {
        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/register.html");
        strncpy(m_real_file + len, m_url_real, strlen(m_url_real));

        free(m_url_real);
    }
    else if (*(p + 1) == '1')
    {
        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/log.html");
        strncpy(m_real_file + len, m_url_real, strlen(m_url_real));

        free(m_url_real);
    }
    else if (*(p + 1) == '5')
    {
        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/picture.html");
        strncpy(m_real_file + len, m_url_real, strlen(m_url_real));

        free(m_url_real);
    }
    else if (*(p + 1) == '6')
    {
        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/video.html");
        strncpy(m_real_file + len, m_url_real, strlen(m_url_real));

        free(m_url_real);
    }
    else if (*(p + 1) == '7')
    {
        char *m_url_real = (char *)malloc(sizeof(char) * 200);
        strcpy(m_url_real, "/fans.html");
        strncpy(m_real_file + len, m_url_real, strlen(m_url_real));

        free(m_url_real);
    }
    else
        strncpy(m_real_file + len, m_url, FILENAME_LEN - len - 1);

    if (stat(m_real_file, &m_file_stat) < 0)
        return NO_RESOURCE;

    if (!(m_file_stat.st_mode & S_IROTH))
        return FORBIDDEN_REQUEST;

    if (S_ISDIR(m_file_stat.st_mode))
        return BAD_REQUEST;

    int fd = open(m_real_file, O_RDONLY);
    m_file_address = (char *)mmap(0, m_file_stat.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
    close(fd);
    return FILE_REQUEST;
}

解除内存映射函数:unmap

void http_conn::unmap()
{
    if (m_file_address)
    {
        munmap(m_file_address, m_file_stat.st_size);// 解除内存映射
        m_file_address = 0;// 清空文件地址
    }
}

发送http响应函数:write

bool http_conn::write()
{
    int temp = 0;

    // 如果没有数据要发送,修改 epoll 事件为 EPOLLIN,并初始化连接
    if (bytes_to_send == 0)
    {
        modfd(m_epollfd, m_sockfd, EPOLLIN, m_TRIGMode);  // 修改 epoll 事件为 EPOLLIN(可读)
        init();  // 重置连接状态
        return true;
    }

    while (1)
    {
        // 使用 writev 系统调用将多个缓冲区的数据写入套接字
        temp = writev(m_sockfd, m_iv, m_iv_count);

        if (temp < 0)
        {
            // 如果写入时返回错误并且 errno 为 EAGAIN,表示当前缓冲区不可写(非阻塞)
            if (errno == EAGAIN)
            {
                // 修改 epoll 事件为 EPOLLOUT(可写),表示稍后可以继续写
                modfd(m_epollfd, m_sockfd, EPOLLOUT, m_TRIGMode);
                return true;
            }
            // 如果发生其他错误,解除内存映射并返回失败
            unmap();
            return false;
        }

        // 更新已发送的字节数,并减少剩余要发送的字节数
        bytes_have_send += temp;
        bytes_to_send -= temp;

        // 如果已经发送完了 m_iv[0] 缓冲区的内容
        if (bytes_have_send >= m_iv[0].iov_len)
        {
            // 设置 m_iv[0] 不再需要写入
            m_iv[0].iov_len = 0;
            // 更新 m_iv[1] 为剩余的数据部分
            m_iv[1].iov_base = m_file_address + (bytes_have_send - m_write_idx);
            m_iv[1].iov_len = bytes_to_send;
        }
        else
        {
            // 否则继续写入 m_iv[0] 缓冲区剩余部分
            m_iv[0].iov_base = m_write_buf + bytes_have_send;
            m_iv[0].iov_len = m_iv[0].iov_len - bytes_have_send;
        }

        // 如果所有数据都已经发送完
        if (bytes_to_send <= 0)
        {
            unmap();  // 解除内存映射
            // 修改 epoll 事件为 EPOLLIN(可以继续读取)
            modfd(m_epollfd, m_sockfd, EPOLLIN, m_TRIGMode);

            // 如果连接保持活动状态(HTTP/1.1),则继续处理后续请求
            if (m_linger)
            {
                init();  // 重置连接状态
                return true;
            }
            else
            {
                return false;  // 关闭连接
            }
        }
    }
}

缓冲区添加数据函数:add_response

通过 vsnprintf 处理可变参数,确保不会超出缓冲区的大小限制,并更新 m_write_idx 来记录写入的位置。如果添加成功,返回 true;如果失败(例如缓冲区不足以容纳数据),则返回 false

bool http_conn::add_response(const char *format, ...)
{
    // 检查写缓冲区是否已经满
    if (m_write_idx >= WRITE_BUFFER_SIZE)
        return false;

    va_list arg_list;
    va_start(arg_list, format);
    
    // 使用 vsnprintf 来格式化数据并写入 m_write_buf 中
    int len = vsnprintf(m_write_buf + m_write_idx, WRITE_BUFFER_SIZE - 1 - m_write_idx, format, arg_list);

    // 检查格式化后的长度是否超出了剩余的缓冲区大小
    if (len >= (WRITE_BUFFER_SIZE - 1 - m_write_idx))
    {
        va_end(arg_list);  // 结束可变参数的处理
        return false;  // 缓冲区空间不足,返回 false
    }
    
    // 更新写入数据的索引
    m_write_idx += len;

    va_end(arg_list);  // 结束可变参数的处理

    // 记录日志,记录请求的响应内容
    LOG_INFO("request:%s", m_write_buf);

    return true;
}

构造报文辅助函数合集

//构建并添加 HTTP 响应的状态行
bool http_conn::add_status_line(int status, const char *title)
{
    return add_response("%s %d %s
", "HTTP/1.1", status, title);
}

//添加响应头部
bool http_conn::add_headers(int content_len)
{
    return add_content_length(content_len) && add_linger() &&
           add_blank_line();
}

//添加 Content-Length 响应头,指示响应体的长度
bool http_conn::add_content_length(int content_len)
{
    return add_response("Content-Length:%d
", content_len);
}

//添加 Content-Type 响应头,指示响应体的类型
bool http_conn::add_content_type()
{
    return add_response("Content-Type:%s
", "text/html");
}

//添加 Connection 响应头,指示客户端与服务器之间的连接是否持续
bool http_conn::add_linger()
{
    return add_response("Connection:%s
", (m_linger == true) ? "keep-alive" : "close");
}

//添加一个空行,表示 HTTP 响应头的结束
bool http_conn::add_blank_line()
{
    return add_response("%s", "
");
}

//添加响应体的内容
bool http_conn::add_content(const char *content)
{
    return add_response("%s", content);
}

构造响应报文函数:process_write

  • 根据不同的 HTTP_CODE 返回值(如 INTERNAL_ERROR, BAD_REQUEST, FORBIDDEN_REQUEST, FILE_REQUEST)生成适当的响应内容。
  • 对于错误响应,函数会构建一个包含错误信息的页面并返回相应的 HTTP 状态码。
  • 对于成功的文件请求,函数会构建一个包含文件内容的响应。
  • 该函数通过调用 add_status_line()add_headers()add_content() 等方法来填充响应报文的各个部分,并通过 m_iv 数组分块发送数据。
bool http_conn::process_write(HTTP_CODE ret)
{
    switch (ret)
    {
    case INTERNAL_ERROR://当发生服务器内部错误时(状态码 500)
    {
        add_status_line(500, error_500_title);
        add_headers(strlen(error_500_form));
        if (!add_content(error_500_form))
            return false;
        break;
    }
    case BAD_REQUEST://当请求不合法时(状态码 400):
    {
        add_status_line(404, error_404_title);
        add_headers(strlen(error_404_form));
        if (!add_content(error_404_form))
            return false;
        break;
    }
    case FORBIDDEN_REQUEST://当访问被拒绝时(状态码 403):
    {
        add_status_line(403, error_403_title);
        add_headers(strlen(error_403_form));
        if (!add_content(error_403_form))
            return false;
        break;
    }
    case FILE_REQUEST://当请求成功并且服务器能够提供文件时(状态码 200):
    {
        add_status_line(200, ok_200_title);
        if (m_file_stat.st_size != 0)
        {
            add_headers(m_file_stat.st_size);
            m_iv[0].iov_base = m_write_buf;
            m_iv[0].iov_len = m_write_idx;
            m_iv[1].iov_base = m_file_address;
            m_iv[1].iov_len = m_file_stat.st_size;
            m_iv_count = 2;
            bytes_to_send = m_write_idx + m_file_stat.st_size;
            return true;
        }
        else
        {
            const char *ok_string = "";
            add_headers(strlen(ok_string));
            if (!add_content(ok_string))
                return false;
        }
    }
    default:
        return false;
    }
    m_iv[0].iov_base = m_write_buf;
    m_iv[0].iov_len = m_write_idx;
    m_iv_count = 1;
    bytes_to_send = m_write_idx;
    return true;
}

http处理核心函数:process

  • 函数的工作流程是:首先调用 process_read() 来解析客户端的请求,如果请求还没有完成,就继续等待更多数据;如果请求已经完成,就通过 process_write() 生成响应并通过 EPOLLOUT 事件准备发送。
  • 它在接收到请求后决定是继续读取数据(通过 EPOLLIN)还是准备发送响应(通过 EPOLLOUT
void http_conn::process()
{
    HTTP_CODE read_ret = process_read();// 读取和解析请求
    if (read_ret == NO_REQUEST)//判断是否需要进一步处理请求
    {
        modfd(m_epollfd, m_sockfd, EPOLLIN, m_TRIGMode);
        return;
    }
    bool write_ret = process_write(read_ret);// 处理请求并生成响应
    if (!write_ret)
    {
        close_conn();
    }
    modfd(m_epollfd, m_sockfd, EPOLLOUT, m_TRIGMode);//修改事件类型为 EPOLLOUT,准备发送响应
}

本文地址:https://www.vps345.com/8879.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 网络工程师 网络管理 软考 2024 2024年上半年 下午真题 答案 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 阿里云 网络 网络安全 网络协议 ubuntu ssh RTSP xop RTP RTSPServer 推流 视频 进程 操作系统 进程控制 Ubuntu python MCP Ollama Qwen2.5-coder 离线部署 deepseek 模型联网 API CherryStudio harmonyos 华为 开发语言 前端 javascript typescript 计算机网络 llama 算法 opencv 自然语言处理 神经网络 语言模型 php 数据库 centos oracle 关系型 安全 分布式 jvm 虚拟机 windows 英语 chrome edge macos conda DNS 容器 人工智能 深度学习 计算机视觉 卷积神经网络 开发环境 vscode 笔记 adb 经验分享 tcp/ip ip协议 数据库系统 vue.js spring boot nginx java docker ubuntu 18.04 安装教程 c语言 负载均衡 tomcat USB转串口 CH340 mysql android fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse Deepseek Deepseek-R1 大模型 私有化部署 推理模型 debian PVE word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 大数据 spark hive jenkins gitee kafka 后端 机器学习 数据分析 eureka https java-rabbitmq fstab gcc centos 7 jellyfin nas openvpn server openvpn配置教程 centos安装openvpn c++ 并查集 leetcode json ide ai AI编程 智能路由器 外网访问 内网穿透 端口映射 DeepSeek-R1 API接口 redis apache kylin ollama llm ssh漏洞 ssh9.9p2 CVE-2025-23419 部署 环境变量 asm mcu 单片机 Hyper-V WinRM TrustedHosts gnu Dify 学习 pip 高级IO epoll 科技 个人开发 学习方法 node.js react.js 前端面试题 持续部署 C# MQTTS 双向认证 emqx ue5 vr 嵌入式 linux驱动开发 arm开发 嵌入式硬件 github 计算机外设 电脑 mac 软件需求 指令 oceanbase 传统数据库升级 银行 ip命令 新增网卡 新增IP 启动网卡 svn 多线程 EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 ddos web安全 cursor pytorch transformer c# 多层架构 解耦 Dell R750XS WSL2 上安装 Ubuntu 物联网 iot 信息与通信 集成学习 集成测试 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 微信 微信分享 鸿蒙 Image wxopensdk 华为云 qt QT 5.12.12 QT开发环境 Ubuntu18.04 vnc udp unity Docker Hub docker pull 镜像源 daemon.json Linux k8s kubernetes rabbitmq 媒体 进程优先级 调度队列 进程切换 自动化 WebUI DeepSeek V3 DeepSeek AI大模型 protobuf 序列化和反序列化 安装 oracle fusion oracle中间件 pycharm rust http 架构 Agent LLM CrewAI yum docker-compose docker compose onlyoffice AI rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK matplotlib fonts-noto-cjk YOLO efficientVIT YOLOv8替换主干网络 TOLOv8 游戏程序 ios uni-app maxkb ARG 数据结构 网络药理学 生信 生物信息学 gromacs 分子动力学模拟 MD 动力学模拟 程序员创富 nlp SRS 流媒体 直播 HarmonyOS Next windows 服务器安装 burp suite 抓包 VMware安装mocOS VMware macOS系统安装 unix C语言 .netcore outlook 错误代码2603 无网络连接 2603 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 产品经理 agi microsoft vim ssl Linux PID rust腐蚀 C 进程地址空间 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 大模型技术 本地部署大模型 dubbo 面试 前端框架 prometheus 监控k8s集群 集群内prometheus git elasticsearch CPU 内存 主板 电源 网卡 vue3 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 互信 本地环回 bind 聚类 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 云原生 ip spring cloud compose HCIE 数通 devops golang powerpoint gateway Clion Nova ResharperC++引擎 Centos7 远程开发 Hive环境搭建 hive3环境 Hive远程模式 Flask FastAPI Waitress Gunicorn uWSGI Uvicorn dify 远程连接 rdp 实验 DevEco Studio 虚拟现实 JAVA Java java-ee 私有化 本地部署 vmware 卡死 numpy Alexnet 开源 milvus VMware创建虚拟机 centos-root /dev/mapper yum clean all df -h / du -sh iftop 网络流量监控 创意 社区 etcd 数据安全 RBAC xcode perf 串口服务器 cron crontab日志 flutter 性能优化 jdk intellij-idea 目标检测 MacOS录屏软件 课程设计 思科 yolov5 运维开发 进程信号 实时音视频 实时互动 stm32项目 stm32 bash zotero WebDAV 同步失败 代理模式 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 ui 华为od audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 机器人 链表 策略模式 spring virtualenv bug websocket 缓存 灵办AI sublime text 编辑器 mybatis TCP WebServer 统信UOS 麒麟 bonding 链路聚合 KylinV10 麒麟操作系统 Vmware maven jar UOS 统信操作系统 深度优先 图论 并集查找 换根法 树上倍增 mq rocketmq rpc AI-native Docker Desktop open webui tcp chatgpt llama3 Chatglm 开源大模型 iBMC UltraISO frp 银河麒麟 kylin v10 麒麟 v10 LDAP deepseek-r1 大模型本地部署 ffmpeg 音视频 多线程服务器 Linux网络编程 域名服务 DHCP 符号链接 配置 C++ WSL2 Ubuntu22.04 虚拟化 开发人员主页 UEFI Legacy MBR GPT U盘安装操作系统 Linux的权限 flash-attention 报错 intellij idea NFS 程序人生 springboot shell 磁盘监控 腾讯云 温湿度数据上传到服务器 Arduino HTTP 爬虫 数据集 ansible playbook 剧本 银河麒麟服务器操作系统 系统激活 博客 虚拟局域网 Linux 维护模式 Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 burpsuite 安全工具 mac安全工具 burp安装教程 渗透工具 webstorm AIGC gpt log4j edge浏览器 sql KingBase WebVM 自动化运维 流水线 脚本式流水线 镜像下载 freebsd MacMini Mac 迷你主机 mini Apple 强制清理 强制删除 mac废纸篓 kind 智能手机 NAS Termux Samba xml express p2p 系统安全 list AI写作 视频编解码 selenium 测试工具 网络结构图 远程工作 命名管道 客户端与服务端通信 RAGFLOW database postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 docker run 数据卷挂载 交互模式 ESP32 Cline 自动化编程 迁移指南 openwrt 虚拟显示器 远程控制 远程 命令 执行 sshpass 操作 嵌入式系统开发 监控k8s 监控kubernetes axure 富文本编辑器 IIS服务器 IIS性能 日志监控 arm WLAN python2 ubuntu24.04 ros2 moveit 机器人运动 rancher oneapi 电子信息 通信工程 毕业 Ubuntu共享文件夹 共享目录 Linux共享文件夹 wireshark 显示过滤器 ICMP Wireshark安装 华为认证 交换机 pygame linux环境变量 minicom 串口调试工具 工业4.0 export import save load 迁移镜像 rime pillow 低代码 区块链 Cursor 客户端 kamailio sip VoIP live555 rtsp rtp 系统开发 binder 车载系统 framework 源码环境 nftables 防火墙 .net WSL win11 无法解析服务器的名称或地址 linux内核 sqlserver 5G 3GPP 卫星通信 豆瓣 追剧助手 迅雷 安卓模拟器 无人机 iperf3 带宽测试 html 浪潮信息 AI服务器 aws googlecloud vSphere vCenter 软件定义数据中心 sddc visual studio code gitlab alias unalias 别名 postgresql ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 僵尸进程 selete 大语言模型 LLMs jmeter 软件测试 webrtc 政务 分布式系统 监控运维 Prometheus Grafana VMware安装Ubuntu Ubuntu安装k8s FTP 服务器 深度求索 私域 知识库 tcpdump 系统架构 微服务 设计模式 软件工程 环境配置 监控 安防软件 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 大数据平台 基础环境 实战案例 Redis Desktop 主从复制 cpu 实时 使用 1024程序员节 交互 html5 firefox 腾讯云大模型知识引擎 数据挖掘 网络用户购物行为分析可视化平台 大数据毕业设计 王者荣耀 云桌面 微软 AD域控 证书服务器 向日葵 TrueLicense docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 驱动开发 硬件工程 嵌入式实习 ipython Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 jupyter linux上传下载 Kali Linux 黑客 渗透测试 信息收集 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos YOLOv8 NPU Atlas800 A300I pro asi_bench django flask web3.py Chatbox Ubuntu Server Ubuntu 22.04.5 ux 软件构建 YOLOv12 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 代码调试 ipdb gpu算力 源码剖析 rtsp实现步骤 流媒体开发 cuda CLion IDE 中间件 iis MQTT协议 消息服务器 代码 ESXi 计算生物学 生物信息 基因组 视觉检测 dell服务器 sqlite3 3d 数学建模 IM即时通讯 QQ 企业微信 剪切板对通 HTML FORMAT 模拟实现 seatunnel 远程看看 远程协助 threejs 3D mongodb okhttp 目标跟踪 OpenVINO 推理应用 rag ragflow ragflow 源码启动 文件系统 路径解析 IPMI 云原生开发 K8S k8s管理系统 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer ue4 着色器 虚幻 DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 Invalid Host allowedHosts vue 开发 cudnn anaconda Reactor rc.local 开机自启 systemd camera Arduino 程序 编程 性能分析 网易邮箱大师 串口驱动 CH341 uart 485 OpenManus Google pay Apple pay 压测 ECS ping++ grafana GCC aarch64 编译安装 HPC 微信小程序 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 HiCar CarLife+ CarPlay QT RK3588 ubuntu20.04 ros ros1 Noetic 20.04 apt 安装 bcompare Beyond Compare windows日志 生活 ollama下载加速 firewalld 读写锁 Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 隐藏文件 隐藏目录 管理器 通配符 linux安装配置 hadoop 话题通信 服务通信 kali 共享文件夹 Windsurf 设置代理 实用教程 tar 大模型部署 离线部署dify 键盘 go 中兴光猫 换光猫 网络桥接 自己换光猫 权限 file server http server web server remote-ssh 安卓 fd 文件描述符 蓝耘科技 元生代平台工作流 ComfyUI 飞牛nas fnos fpga开发 本地化部署 鸿蒙系统 springcloud gradle Linux权限 权限命令 特殊权限 DeepSeek r1 Open WebUI 硬件架构 计算机 conda配置 conda镜像源 职场和发展 远程过程调用 Windows环境 执法记录仪 智能安全帽 smarteye 图形化界面 换源 国内源 Debian 硬件 设备 GPU PCI-Express SysBench 基准测试 服务器时间 webdav ecm bpm 游戏服务器 Minecraft 安全架构 安全威胁分析 crosstool-ng Erlang OTP gen_server 热代码交换 事务语义 MAC SecureCRT 远程桌面 MNN Qwen minio 免费域名 域名解析 HP Anyware wsl2 wsl x64 SIGSEGV SSE xmm0 eclipse 热榜 稳定性 看门狗 ci/cd es6 qt6.3 g726 web Typore 其他 tensorflow trae CentOS gaussdb redhat 具身智能 Isaac Sim 虚拟仿真 服务器繁忙 宝塔面板 ruoyi Dell HPE 联想 浪潮 iDRAC R720xd 跨域 边缘计算 npm pdf 办公自动化 自动化生成 pdf教程 vpn 分析解读 服务器扩容没有扩容成功 交叉编译 压力测试 测试用例 功能测试 asp.net大文件上传 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 ArkTs ArkUI elk vasp安装 内网渗透 靶机渗透 AI作画 服务器配置 arcgis 聊天室 AutoDL 图像处理 zabbix 视频平台 录像 视频转发 性能测试 视频流 存储 GameFramework HybridCLR Unity编辑器扩展 自动化工具 矩阵 Mac内存不够用怎么办 ELF加载 pyautogui VPN wireguard muduo zip unzip 信号 X11 Xming 软链接 硬链接 android studio ebpf uprobe Trae AI 原生集成开发环境 Trae AI 群晖 飞牛 小艺 Pura X mariadb excel 代理 硅基流动 ChatBox kvm qemu libvirt AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 prompt 网络爬虫 bot Docker 宠物 毕业设计 免费学习 宠物领养 宠物平台 推荐算法 CosyVoice Xinference RAGFlow 自动驾驶 显示管理器 lightdm gdm c 华为证书 HarmonyOS认证 华为证书考试 opengl 电视剧收视率分析与可视化平台 cd 目录切换 vscode 1.86 网站搭建 serv00 jetty undertow 文心一言 grub 版本升级 扩容 远程登录 telnet 大模型入门 大模型教程 SSH Xterminal 信号处理 VR手套 数据手套 动捕手套 动捕数据手套 ubuntu24 vivado24 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 fork wait waitpid exit banner su sudo Ark-TS语言 apt 单例模式 ROS PX4 MAVROS 四旋翼无人机 virtualbox DocFlow 雨云 NPS SWAT 配置文件 服务管理 网络共享 cmos 孤岛惊魂4 输入法 llama.cpp 程序员 DeepSeek行业应用 Heroku 网站部署 xpath定位元素 k8s集群资源管理 Web服务器 多线程下载工具 网络编程 PYTHON mamba 游戏机 工具 hugo Netty 即时通信 NIO av1 电视盒子 机顶盒ROM 魔百盒刷机 keepalived 大模型微调 双系统 kotlin iphone 思科模拟器 Cisco sonoma 自动更新 nuxt3 毕设 Helm k8s集群 lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 neo4j 数据仓库 数据库开发 数据库架构 vmamba MQTT 消息队列 技能大赛 重启 排查 系统重启 日志 原因 半虚拟化 硬件虚拟化 Hypervisor opcua opcda KEPServer安装 micropython esp32 mqtt mosquitto ArcTS 登录 ArcUI GridItem arkUI r语言 数据可视化 实习 算力 服务器管理 配置教程 服务器安装 网站管理 高效I/O Mermaid 可视化图表 sqlite 内核 n8n dity make chrome devtools chromedriver openssl 密码学 HarmonyOS NEXT 原生鸿蒙 MS Materials 模拟退火算法 curl wget copilot 田俊楠 Kali langchain deep learning 统信 UOS1070e visualstudio VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 搜索引擎 searxng 代码托管服务 clickhouse 安装MySQL 社交电子 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 hibernate css HarmonyOS OpenHarmony W5500 OLED u8g2 TCP服务器 SEO chfs ubuntu 16.04 本地部署AI大模型 漏洞 同步 备份 建站 框架搭建 Reactor反应堆 firewall Zoertier 内网组网 回显服务器 UDP的API使用 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 金仓数据库 2025 征文 数据库平替用金仓 直流充电桩 充电桩 Docker Compose 微信开放平台 微信公众平台 微信公众号配置 docker命令大全 系统 黑苹果 小游戏 五子棋 nac 802.1 portal nacos sdkman EasyConnect openEuler MCP server C/S unity3d TRAE RustDesk自建服务器 rustdesk服务器 docker rustdesk 网络穿透 自动化测试 火绒安全 图形渲染 Nuxt.js uniapp nvidia ssrf 失效的访问控制 Windows 恒源云 lvm 磁盘挂载 磁盘分区 rustdesk 相机 毕昇JDK 小智 致远OA OA服务器 服务器磁盘扩容 CentOS Stream 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 OD机试真题 华为OD机试真题 服务器能耗统计 xfce CORS 飞书 dns 蓝桥杯 xrdp ceph nvm 混合开发 环境安装 JDK matlab 网络攻击模型 SSL证书 lb 协议 scikit-learn 强化学习 能力提升 面试宝典 技术 IT信息化 Web应用服务器 mysql离线安装 ubuntu22.04 mysql8.0 源码 网页服务器 web服务器 Nginx code-server webgl 信息可视化 echarts gpt-3 XCC Lenovo Ubuntu 24.04.1 轻量级服务器 ArkTS 移动端开发 考研 在线office 7z 京东云 移动云 CDN pgpool eNSP 网络规划 VLAN 企业网络 kernel 大模型面经 大模型学习 VSCode 显卡驱动 业界资讯 AnythingLLM AnythingLLM安装 命令行 基础入门 鲲鹏 端口测试 odoo 服务器动作 Server action chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 容器技术 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm Linux Vim wsgiref Web 服务器网关接口 联网 easyconnect nfs 网页设计 序列化反序列化 jina SSL 域名 skynet MDK 嵌入式开发工具 论文笔记 快捷键 旋转屏幕 自动操作 人工智能生成内容 金融 AISphereButler embedding ROS2 Python 国产操作系统 ukui 麒麟kylinos openeuler 拓扑图 top Linux top top命令详解 top命令重点 top常用参数 虚拟机安装 can 线程池 浏览器自动化 rclone AList fnOS ruby Carla 智能驾驶 java-rocketmq 游戏引擎 Jellyfin 线程 LORA NLP cfssl MySql ssh远程登录 P2P HDLC 宝塔 飞牛NAS 飞牛OS MacBook Pro Kylin-Server harmonyOS面试题 健康医疗 互联网医院 cnn GRUB引导 Linux技巧 邮件APP 免费软件 宝塔面板无法访问 开机黑屏 Linux的基础指令 黑客技术 大屏端 URL springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 Playwright VPS springsecurity6 oauth2 授权服务器 前后端分离 pyqt MI300x WebRTC RAID RAID技术 磁盘 samba uv 昇腾 npu open Euler dde deepin RTMP 应用层 safari vite big data 历史版本 下载 opensearch helm 服务器主板 AI芯片 etl Linux24.04 c/c++ 串口 string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap TrinityCore 魔兽世界 多进程 sysctl.conf vm.nr_hugepages AI代码编辑器 adobe 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 vu大文件秒传跨域报错cors 聊天服务器 套接字 Socket wps 智能音箱 智能家居 IPMITOOL BMC 硬件管理 yaml Ultralytics 可视化 stable diffusion ecmascript KVM python3.11 make命令 makefile文件 asp.net大文件上传下载 dash 正则表达式 文件分享 繁忙 解决办法 替代网站 汇总推荐 AI推理 云服务 雨云服务器 可信计算技术 rnn token sas dba 崖山数据库 YashanDB ocr 银河麒麟操作系统 国产化 服务器部署ai模型 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 IIS .net core Hosting Bundle .NET Framework vs2022 镜像 rsyslog 机柜 1U 2U Anolis nginx安装 linux插件下载 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 用户缓冲区 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 代码规范 saltstack 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 支付 微信支付 开放平台 Cookie 无桌面 risc-v glibc SenseVoice DBeaver composer 开机自启动 产测工具框架 IMX6ULL 管理框架 三级等保 服务器审计日志备份 k8s二次开发 集群管理 cocoapods openstack Xen 国产数据库 瀚高数据库 数据迁移 下载安装 云耀服务器 内网环境 bootstrap 版本 Tabs组件 TabContent TabBar TabsController 导航页签栏 滚动导航栏 Attention nextjs react reactjs visual studio Vmamba yolov8 流式接口 api 渗透 React Next.js 开源框架 联想开天P90Z装win10 宕机切换 服务器宕机 lio-sam SLAM less 云电竞 云电脑 todesk DIFY 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? 大文件秒传跨域报错cors 网工 RAG 检索增强生成 文档解析 大模型垂直应用 自学笔记 小米 澎湃OS Android DOIT 四博智联 Unity Dedicated Server Host Client 无头主机 医疗APP开发 app开发 数据管理 数据治理 数据编织 数据虚拟化 trea idea Node-Red 编程工具 流编程 idm 模拟器 教程 ubuntu24.04.1 NLP模型 树莓派 VNC 匿名管道 db fast notepad bat 大模型应用 thingsboard 工作流 workflow OpenSSH 图片增强 增强数据 powerbi IO 环境迁移 开源软件 XFS xfs文件系统损坏 I_O error 直播推流 es ftp服务 文件上传 IPv4 子网掩码 公网IP 私有IP SSH 密钥生成 SSH 公钥 私钥 生成 相差8小时 UTC 时间 next.js 部署next.js netty Qualcomm WoS QNN AppBuilder linux 命令 sed 命令 SSH 服务 SSH Server OpenSSH Server gitea 智能电视 k8s部署 MySQL8.0 高可用集群(1主2从) ShenTong FunASR ASR 自动化任务管理 easyui 状态管理的 UDP 服务器 Arduino RTOS 智慧农业 开源鸿蒙 团队开发 磁盘清理 Wi-Fi 跨平台 Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 yum源切换 更换国内yum源 单元测试 ISO镜像作为本地源 远程服务 retry 重试机制 AzureDataStudio 元服务 应用上架 qps 高并发 线性代数 电商平台 pyicu 抓包工具 阻塞队列 生产者消费者模型 服务器崩坏原因 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 数码 联机 僵尸毁灭工程 游戏联机 开服 软件 欧标 OCPP 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 lua 音乐服务器 Navidrome 音流 qt5 客户端开发 AI员工 大版本升 升级Ubuntu系统 dns是什么 如何设置电脑dns dns应该如何设置 perl 笔灵AI AI工具 李心怡 端口号 开放端口 访问列表 查看显卡进程 fuser ArtTS ECT转485串口服务器 ECT转Modbus485协议 ECT转Modbus串口服务器 IO模型 deekseek 音乐库 Ardupilot 影刀 #影刀RPA# sublime text3 g++ g++13 docker部署Python 环境搭建 Maven Office MacOS ShapeFile GeoJSON AI agent iventoy VmWare OpenEuler uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 Linux环境 Radius 运维监控 Qwen2.5-VL vllm DenseNet cmake 物联网开发 figma EMQX 通信协议 计算虚拟化 弹性裸金属 游戏开发 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR GoogLeNet 负载测试 nosql 做raid 装系统 问题解决 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 冯诺依曼体系 MobaXterm 文件传输 虚幻引擎 hosts HTTP 服务器控制 ESP32 DeepSeek 银河麒麟桌面操作系统 Kylin OS xss AD域 win服务器架设 windows server Sealos env 变量 我的世界服务器搭建 minecraft 论文阅读 空间 查错 软件卸载 系统清理 烟花代码 烟花 元旦 本地知识库部署 DeepSeek R1 模型 蓝牙 xshell termius iterm2 查询数据库服务IP地址 SQL Server 分布式训练 语音识别 进程间通信 实时内核 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 nohup后台启动 服务网格 istio 状态模式 csrutil mac恢复模式进入方法 SIP 恢复模式 js 授时服务 北斗授时 免密 公钥 私钥 grep FTP服务器 qt项目 qt项目实战 qt教程 网络建设与运维 网络搭建 神州数码 神州数码云平台 云平台 国标28181 视频监控 监控接入 语音广播 流程 SDP sudo原理 su切换 银河麒麟高级服务器 外接硬盘 Kylin 极限编程 反向代理 VS Code 合成模型 扩散模型 图像生成 根服务器 PPI String Cytoscape CytoHubba 华为机试 umeditor粘贴word ueditor粘贴word ueditor复制word ueditor上传word图片 ueditor导入word ueditor导入pdf ueditor导入ppt 知识图谱 ufw 自定义客户端 SAS junit 抗锯齿 hexo 多产物 scapy LInux 鸿蒙开发 移动开发 小智AI服务端 xiaozhi TTS 语法 AD 域管理 iNode Macos laravel 服务器部署 本地拉取打包 armbian u-boot ftp 网络文件系统 lighttpd安装 Ubuntu配置 Windows安装 服务器优化 升级 CVE-2024-7347 需求分析 规格说明书 sequoiaDB react native deepseak 豆包 KIMI 腾讯元宝 CNNs 图像分类 捆绑 链接 谷歌浏览器 youtube google gmail 进程程序替换 execl函数 execv函数 execvp函数 execvpe函数 putenv函数 内网服务器 内网代理 内网通信 MVS 海康威视相机 prometheus数据采集 prometheus数据模型 prometheus特点 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 备选 网站 调用 示例 vscode1.86 1.86版本 ssh远程连接 ranger MySQL8.0 LLM Web APP Streamlit proxy模式 大模型训练/推理 推理问题 mindie 性能调优 安全代理 桌面环境 regedit 开机启动 北亚数据恢复 oracle数据恢复 dock 加速 Python基础 Python教程 Python技巧 whistle Ubuntu 22.04 算家云 算力租赁 diskgenius Headless Linux CPU 使用率 系统监控工具 linux 命令 HistoryServer Spark YARN jobhistory wpf Claude 玩游戏 Apache Beam 批流统一 案例展示 数据分区 容错机制 mcp服务器 client close flink WINCC 玩机技巧 软件分享 软件图标 ajax netlink libnl3 沙盒 beautifulsoup word 底层实现 多路转接 大大通 第三代半导体 碳化硅 Mac软件 ai工具 v10 移动魔百盒 对比 meld DiffMerge 一切皆文件 终端工具 远程工具 h.264 USB网络共享 项目部署到linux服务器 项目部署过程 超融合 EVE-NG 工具分享 shell脚本免交互 expect linux免交互 web3 Xshell 浏览器开发 AI浏览器 项目部署 pppoe radius 华为OD 可以组成网络的服务器 网卡的名称修改 eth0 ens33 cpp-httplib NAT转发 NAT Server 单一职责原则 deepseek r1 IMM 常用命令 文本命令 目录命令 tidb GLIBC deployment daemonset statefulset cronjob kerberos zookeeper nohup 异步执行 僵尸世界大战 游戏服务器搭建 IMX317 MIPI H265 VCU GIS 遥感 WebGIS TCP协议 计算机科学与技术 navicat swoole 阿里云ECS ldap yashandb seleium 多个客户端访问 IO多路复用 TCP相关API 端口聚合 windows11 达梦 DM8 钉钉 架构与原理 Logstash 日志采集 IDEA gunicorn tailscale derp derper 中转 triton 模型分析 System V共享内存 进程通信 软负载 docker desktop image iTerm2 高德地图 鸿蒙接入高德地图 HarmonyOS5.0 vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 端口 查看 ss 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 H3C 信创 信创终端 中科方德 sentinel 搭建个人相关服务器 midjourney 嵌入式Linux IPC ollama api ollama外网访问 docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 日志分析 系统取证 EMUI 回退 降级 企业网络规划 华为eNSP 代理服务器 佛山戴尔服务器维修 佛山三水服务器维修 多端开发 智慧分发 应用生态 鸿蒙OS wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 rpa 个人博客 加解密 Yakit yaklang Docker快速入门 干货分享 黑客工具 密码爆破 Spring Security 技术共享 我的世界 我的世界联机 Linux find grep glm4 DevOps 软件交付 数据驱动 mysql安装报错 windows拒绝安装 css3 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 带外管理 流量运营 解决方案 服务器正确解析请求体 子系统 管道 信息安全 Unity插件 initramfs Linux内核 Grub SVN Server tortoise svn 风扇控制软件 yum换源 蓝桥杯C++组 VGG网络 卷积层 池化层 LVM lvresize 磁盘扩容 pvcreate SystemV 考试 动静态库 mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 7-zip Ubuntu20.04 2.35 大模型推理 resolv.conf 脚本 autodl Echarts图表 折线图 柱状图 异步动态数据 可视化效果 fiddler 行情服务器 股票交易 速度慢 切换 股票量化接口 股票API接口 全文检索 图搜索算法 solidworks安装 零售 智能合约 哈希算法 显示器 电路仿真 multisim 硬件工程师 硬件工程师学习 电路图 电路分析 仪器仪表 终端 pyside6 界面 HAProxy ABAP postgres Dify重启后重新初始化 termux 网络原理 弹性服务器 chromium dpi 程序化交易 量化交易 高频交易 macOS 服务器ssl异常解决 配置原理 输入系统 分布式账本 信任链 共识算法 openjdk IP配置 netplan 智能体开发 archlinux kde plasma 存储维护 NetApp存储 EMC存储 ardunio BLE 安全漏洞 Pyppeteer llamafactory 微调 西门子PLC 通讯 电脑桌面出现linux图标 电脑桌面linux图标删除不了 电脑桌面Liunx图标删不掉 linux图标删不掉 新盘添加 partedUtil GRE linuxdeployqt 打包部署程序 appimagetool 粘包问题 webview 用户管理 搜狗输入法 中文输入法 打不开xxx软件 无法检查其是否包含恶意软件 框架 arkTs autoware UDP 充电桩平台 充电桩开源平台 java-zookeeper GPU训练 zerotier 接口优化 Claude Desktop Claude MCP Windows Cli MCP 锁屏不生效 NFC 近场通讯 智能门锁 AI Agent 字节智能运维 内存管理 显卡驱动持久化 GPU持久化 数字证书 签署证书 WireGuard 异地组网 AimRT GeneCards OMIM TTD 分子对接 autodock mgltools PDB PubChem 桌面快捷方式 IPv6 IPv6测试 IPv6测速 IPv6检测 IPv6查询 massa sui aptos sei 星河版 机械臂 rtc 券商 股票交易接口api 类型 特点 金仓数据库概述 金仓数据库的产品优化提案 IPv4/IPv6双栈 双栈技术 网路规划设计 ensp综合实验 IPv4过渡IPv6 IPv4与IPv6 区块链项目 物理地址 页表 虚拟地址 PyQt PySide6 localhost 三次握手 pthread Webserver 异步 servlet Python 视频爬取教程 Python 视频爬取 Python 视频教程 设备树 hdc 鸿蒙NEXT macbook 支持向量机 文件存储服务器组件 ubuntu安装 linux入门小白 动态规划 商用密码产品体系 NVM Node Yarn PM2 Bug解决 Qt platform OpenCV RDP 软件开发 logstash 免费 post.io 企业邮箱 搭建邮箱 kubeless c/s 跨域请求 设计规范 element-ui 上传视频并预览视频 vue上传本地视频及进度条功能 vue2选择视频上传到服务器 upload上传视频组件插件 批量上传视频 限制单个上传视频 飞腾处理器 海康 NVML nvidia-smi 源代码管理 #STC8 #STM32