• 【ESP32】ESP-IDF开发 | WiFi开发 | TCP传输控制协议 + TCP服务器和客户端例程

【ESP32】ESP-IDF开发 | WiFi开发 | TCP传输控制协议 + TCP服务器和客户端例程

2025-04-26 05:51:13 2 阅读

1. 简介

        TCP(Transmission Control Protocol),全称传输控制协议。它的特点有以下几点:面向连接,每一个TCP连接只能是点对点的(一对一);提供可靠交付服务;提供全双工通信面向字节流

1.1 三次握手

        三次握手代表的是TCP的连接过程,它表示在成功连接之前,服务端和客户端需要通信三次才能最终确认。

  • 第一次握手:客户端将标志位SYN置为1,随机产生一个值seq=J,并将该数据包发送给服务器端,客户端进入SYN_SENT状态,等待服务器端确认;
  • 第二次握手:服务器端收到数据包后由标志位SYN=1知道客户端请求建立连接,服务器端将标志位SYN和ACK都置为1,ack=J+1,随机产生一个值seq=K,并将该数据包发送给客户端以确认连接请求,服务器端进入SYN_RCVD状态;
  •  第三次握手:客户端收到确认后,检查ack是否为J+1,ACK是否为1,如果正确则将标志位ACK置为1,ack=K+1,并将该数据包发送给服务器端,服务器端检查ack是否为K+1,ACK是否为1,如果正确则连接建立成功,客户端和服务器端进入ESTABLISHED状态,完成三次握手,随后客户端与服务器端之间可以开始传输数据了。

1.2 四次挥手

        四次挥手代表的是TCP的断开连接过程,它表示在成功断开前,服务端和客户端需要通信四次才能最终确认。

  • 第一次挥手:客户端发送一个FIN=M,用来关闭客户端到服务器端的数据传送,客户端进入FIN_WAIT_1状态,表示客户端没有数据需要发送了;但是如果服务器端还有数据没有发送完成,则可以继续发送数据;
  • 第二次挥手:服务器端收到FIN后,先发送ack=M+1,告诉客户端请求收到了,但是我还没准备好,要继续等待我的消息;这个时候客户端就进入FIN_WAIT_2 状态,继续等待服务器端的FIN报文;
  • 第三次挥手:当服务器端确定数据已发送完成,则向客户端发送FIN=N报文,告诉客户端数据发完了,准备关闭连接;服务器端进入LAST_ACK状态;
  • 第四次挥手:客户端收到FIN=N报文后,就知道可以关闭连接了,但是他还是不相信网络,怕服务器端不知道要关闭,所以发送ack=N+1后会进入TIME_WAIT状态,如果服务端没有收到ACK则可以重传。服务器端收到ACK后,就知道可以断开连接了。如果客户端等待了2MSL后依然没有收到回复,则证明服务器端已正常关闭,那客户端也可以关闭连接了。

 

1.3 拥塞控制

         网络就像我们生活中的交通系统,当车流大的时候就可能会导致拥塞,TCP为了保证可靠的交付服务,所以引入了拥塞控制,灵活调整发送策略。

        拥塞控制是一个动态的过程,它既要提高带宽利用率发送尽量多的数据又要避免网络拥堵丢包RTT增大等问题,基于这种高要求并不是单一策略可以搞定的,因此TCP的拥塞控制策略实际上是分阶段分策略的综合过程,包括慢开始(slow start)、拥塞避免(congestion avoidance)、快重传(fast retransmit)和快恢复(fast recovery)。

1. 慢开始

        慢开始算法的思路为,在数据开始发送时,由于不清楚网络的负荷情况,如果此时立即把大量数据发送到网络,那么就有可能引起网络拥塞。根据生活中的经验进行引伸,较好的方法是由小大到逐渐增大发送窗口,一步步探测网络链路的极限;也就是说,由小到大逐渐增大拥塞窗口数值(cwnd),下面是其简要工作流程图。

        由上图可见,一开始发送方的初始cwnd 为1,发送方发送第一个报文段M1,并收到接收方的确认。此时,发送方将cwnd从1增大到2,接着发送M2和M3两个报文段,收到两个报文的确认后,发送方继续将cwnd加倍,增加到4。只要网络仍然通畅,那么发送方就会以此类推,在每一轮的传输成功后将cwnd进行加倍的操作。

        显然,cwnd不能无限制地加倍,这样会引起网络拥塞,因此需要设置一个慢开始门限(ssthresh)状态变量。当cwnd < ssthresh时,才会使用上述的慢开始算法。

2. 拥塞避免

        当cwnd > ssthresh时,系统会使用拥塞避免算法,该算法的思路是让拥塞窗口(cwnd)缓慢地增长,即每完成一轮传输就把发送方的拥塞窗口(cwnd)加1,而不是像慢开始阶段那样加倍增长。因此在拥塞避免阶段就有“加法增大”(Additive Increase)的特点。这表明在拥塞避免阶段,拥塞窗口(cwnd)按线性规律缓慢增长,比慢开始算法的拥塞窗口增长速率缓慢得多。下图展示了拥塞控制的工作流程。

        在上图中,慢开始门限的初始值为16。一开始系统执行慢开始算法,发送方每成功发送一轮报文段,就把拥塞窗口值加倍,然后开始下一轮的传输。因此拥塞窗口cwnd随着传输轮次按指数规律增长。当拥塞窗口cwnd增长到慢开始门限值ssthresh 时,就开始改为执行拥塞避免算法,拥塞窗口按线性规律增长。

        当进行到第12轮传输时(上图节点2),网络出现了超时,发送方判断为网络拥塞。于是调整门限值ssthresh = cwnd / 2 = 12,同时设置拥塞窗口cwnd = 1,重新进入慢开始阶段。

        上图节点3展示了一个特殊情况,此时拥塞窗口cwnd = 16,这时出现了发送方一连收到3个对同一报文段的重复确认的情况,此时执行了拥塞避免算法,调整门限值ssthresh = cwnd / 2 = 8。这是因为如果发送方迟迟收不到确认,就会产生超时,会误认为网络发生了拥塞。这就导致发送方错误地启动慢开始,把拥塞窗口cwnd又置为1,因而降低了传输效率。

3. 快重传

        TCP作为一个可靠的协议面临的很大的问题就是丢包,丢包就要重传因此发送方需要根据接收方回复的ACK来确认是否丢包了,下图为超时重传的典型时序图。

        重传超时时间(RTO)是随着复杂网络环境而动态变化的,在拥塞控制中发生超时重传将会极大拉低cwnd,如果网络状况并没有那么多糟糕,偶尔出现网络抖动造成丢包或者阻塞也非常常见,因此触发的慢启动将降低通信性能,故出现了快速重传机制。所谓快速重传时相比超时重传而言的,重发等待时间会降低并且后续尽量避免慢启动,来保证性能损失在最小的程度,下图为其时序图。

        快速重传和超时重传的区别在于cwnd在发生拥塞时的取值,超时重传会将cwnd修改为最初的值,也就是慢启动的值,快速重传将cwnd减半,二者都将ssthresh设置为cwnd的一半。从二者的区别可以看到,快速重传更加主动,有利于保证链路的传输性能。

4. 快恢复

        在快速重传之后就会进入快速恢复阶段,此时的cwnd为上次发生拥塞时的cwnd的1/2,之后cwnd再线性增加重复之前的过程。

2. lwIP

        ESP-IDF使用lwIP库实现TCP/IP协议栈,这个库在大多数嵌入式系统中都有用到,它是对底层硬件的上层封装,所以如果未来要写比如Linux的TCP/IP应用,代码也是通用的。

3. 例程

        例程分别在ESP32上实现TCP客户端和服务端,使用电脑作为另一方进行简单通信测试。需要注意的是,测试时,ESP32和电脑必须处于同一局域网

        电脑端测试会使用的上位机为野火串口调试助手,下载地址:FireTools

3.1 客户端

        这个例程配置ESP32为客户端,当连接WiFi热点成功后会请求连接服务端,连接成功后会发送一段消息,然后阻塞等待服务端回复,服务端恢复消息后ESP32会主动关闭套接字。

#include "freertos/FreeRTOS.h"
#include "freertos/queue.h"
#include "freertos/semphr.h"
#include "esp_system.h"
#include "esp_wifi.h"
#include "esp_event.h"
#include "esp_log.h"
#include "esp_mac.h"
#include "nvs_flash.h"
#include "sys/socket.h"
#include "lwip/err.h"
#include "lwip/sys.h"
#include "netdb.h"
#include "arpa/inet.h"

#include 

#define TAG "app"
#define HOST_IP_ADDR "192.168.10.117"
#define HOST_PORT 20001


static char rx_buffer[128];
static const char *payload = "Message from ESP32";
static TaskHandle_t client_task_handle;

static void tcp_client_task(void *args)
{
    struct sockaddr_in dest_addr;
    inet_pton(AF_INET, HOST_IP_ADDR, &dest_addr.sin_addr);
    dest_addr.sin_family = AF_INET;
    dest_addr.sin_port = htons(HOST_PORT);

    while (1) {
        int sock =  socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
        if (sock < 0) {
            ESP_LOGE(TAG, "Unable to create socket: errno %d", errno);
            break;
        }
        ESP_LOGI(TAG, "Socket created, connecting to %s:%d", HOST_IP_ADDR, HOST_PORT);

        int err = connect(sock, (struct sockaddr *)&dest_addr, sizeof(dest_addr));
        if (err != 0) {
            ESP_LOGE(TAG, "Socket unable to connect: errno %d", errno);
            break;
        }
        ESP_LOGI(TAG, "Successfully connected");

        err = send(sock, payload, strlen(payload), 0);
        if (err < 0) {
            ESP_LOGE(TAG, "Error occurred during sending: errno %d", errno);
            break;
        }

        memset(rx_buffer, 0, sizeof(rx_buffer));
        int len = recv(sock, rx_buffer, sizeof(rx_buffer) - 1, 0);
        if (len < 0) {
            ESP_LOGE(TAG, "recv failed: errno %d", errno);
        } else {
            ESP_LOGI(TAG, "Received %d bytes from %s:", len, HOST_IP_ADDR);
            ESP_LOGI(TAG, "data: %s", rx_buffer);
        }

        close(sock);

        vTaskDelay(1000 / portTICK_PERIOD_MS);
    }
}

static void wifi_event_handler(void* arg,
                               esp_event_base_t event_base,
                               int32_t event_id,
                               void* event_data)
{
    if (event_base == IP_EVENT) {
        if (event_id == IP_EVENT_STA_GOT_IP) {
            xTaskCreate(tcp_client_task, "tcp_client", 2048, NULL, 5, &client_task_handle);
        }
    } else if (event_base == WIFI_EVENT) {
        if (event_id == WIFI_EVENT_STA_DISCONNECTED) {
            vTaskDelete(client_task_handle);
        } else if (event_id == WIFI_EVENT_STA_START) {
            esp_wifi_connect();
        }
    }
}

int app_main()
{
    /* 初始化NVS */
    esp_err_t ret = nvs_flash_init();
    if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) {
        ESP_ERROR_CHECK(nvs_flash_erase());
        ESP_ERROR_CHECK(nvs_flash_init());
    }

    /* 初始化WiFi协议栈 */
    ESP_ERROR_CHECK(esp_netif_init());
    ESP_ERROR_CHECK(esp_event_loop_create_default());
    esp_netif_create_default_wifi_sta();

    wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
    ESP_ERROR_CHECK(esp_wifi_init(&cfg));

    ESP_ERROR_CHECK(esp_event_handler_instance_register(WIFI_EVENT,
                                                        ESP_EVENT_ANY_ID,
                                                        &wifi_event_handler,
                                                        NULL,
                                                        NULL));

    ESP_ERROR_CHECK(esp_event_handler_instance_register(IP_EVENT,
                                                        ESP_EVENT_ANY_ID,
                                                        &wifi_event_handler,
                                                        NULL,
                                                        NULL));

    wifi_config_t wifi_config = {
        .sta = {
            .ssid = "Your SSID",
            .password = "Your password",
            .threshold.authmode = WIFI_AUTH_WPA_WPA2_PSK,
        },
    };

    ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA));
    ESP_ERROR_CHECK(esp_wifi_set_config(WIFI_IF_STA, &wifi_config));
    ESP_ERROR_CHECK(esp_wifi_start());

    return 0;
}

        ESP32的WiFi驱动初始化在前面的文章已经有详细的介绍了,这里不再赘述。

        在回调函数中,当驱动获取到IP后,就会创建TCP客户端的任务。

1. 创建socket套接字

        调用socket函数创建,第一个参数表示域,这里使用IPv4,对应IP_INET;第二个参数表示socket类型,TCP协议只能填SOCK_STREAM;第三个参数表示协议栈类型,这里填IPPROTO_IP。函数会返回套接字描述符。

2. 连接服务器

        调用connect函数,第一个参数传入套接字描述符;比较重要的是第二个参数,要传入服务器的地址信息。结构体的定义如下:

struct sockaddr_in {
  u8_t            sin_len;
  sa_family_t     sin_family;
  in_port_t       sin_port;
  struct in_addr  sin_addr;
#define SIN_ZERO_LEN 8
  char            sin_zero[SIN_ZERO_LEN];
};
  • sin_len:数据长度(一般不需要填);
  • sin_family:套接字类型,IPv4填AF_INET,IPv6填AF_INET6,其他填AF_UNSPEC;
  • sin_port:端口;
  • sin_zero:上层预留字节(不用管)。

3.  发送数据

        调用send函数。传入套接字描述符、数据指针和数据长度即可;最后一个参数是标志位,一般填0即可,可选的标志位如下:

#define MSG_PEEK       0x01
#define MSG_WAITALL    0x02
#define MSG_OOB        0x04
#define MSG_DONTWAIT   0x08
#define MSG_MORE       0x10
#define MSG_NOSIGNAL   0x20

        这些标志位是发送和接收都支持的,比较常用的是MSG_DONTWAIT,像发送和接收函数是阻塞的,使能这个标志位可以让函数立即返回,不等待数据。

4. 接收数据

        调用recv函数。传入的参数与send函数是一致的,不再赘述。

5. 关闭连接

        调用close函数。传入套接字描述符即可。

        测试的时候先打开上位机,设置为TCP服务器,填写电脑的IP和端口,端口是自定义的,但注意不要与原有的端口冲突,建议设置20000以上比较保险;最后点击开始监听。

3.2 服务端

        这个例程就是在ESP32上搭建一个TCP服务器,接受局域网中的客户端连接并接收数据。

#include "freertos/FreeRTOS.h"
#include "freertos/queue.h"
#include "freertos/semphr.h"
#include "esp_system.h"
#include "esp_wifi.h"
#include "esp_event.h"
#include "esp_log.h"
#include "esp_mac.h"
#include "nvs_flash.h"
#include "sys/socket.h"
#include "lwip/err.h"
#include "lwip/sys.h"
#include "netdb.h"
#include "arpa/inet.h"

#include 

#define TAG "app"
#define HOST_PORT 20001

static const char *payload = "I have received your message";
static TaskHandle_t server_task_handle;


static void tcp_client_task(void *args)
{
    int *sock = args;
    int len;
    char rx_buffer[128] = {0};

    while (1) {
        memset(rx_buffer, 0, sizeof(rx_buffer));
        len = recv(*sock, rx_buffer, sizeof(rx_buffer) - 1, 0);
        if (len < 0) {
            ESP_LOGE(TAG, "Error occurred during receiving: errno %d", errno);
        } else if (len == 0) {
            ESP_LOGW(TAG, "Connection closed");
            goto __exit;
        } else {
            ESP_LOGI(TAG, "Received %d bytes, data: %s", len, rx_buffer);
            send(*sock, payload, strlen(payload), 0);
        }
    }

__exit:
    close(*sock);
    free(sock);
    vTaskDelete(NULL);
}

static void tcp_server_task(void *args)
{
    esp_ip4_addr_t *ip_addr = args;

    struct sockaddr_in dest_addr = {0};
    dest_addr.sin_addr.s_addr = ip_addr->addr;
    dest_addr.sin_family = AF_INET;
    dest_addr.sin_port = htons(HOST_PORT);

    int sock = socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
    if (sock < 0) {
        ESP_LOGE(TAG, "Unable to create socket: errno %d", errno);
        goto __exit;
    }

    int err = bind(sock, (struct sockaddr *)&dest_addr, sizeof(dest_addr));
    if (err != 0) {
        ESP_LOGE(TAG, "Socket unable to bind: errno %d", errno);
        goto __exit;
    }

    err = listen(sock, 1);
    if (err != 0) {
        ESP_LOGE(TAG, "Error occurred during listen: errno %d", errno);
        goto __exit;
    }

    ESP_LOGI(TAG, "Server listen at " IPSTR ":%d", IP2STR(ip_addr), HOST_PORT);

    while (1) {
        struct sockaddr_in source_addr = {0};
        socklen_t addr_len = sizeof(struct sockaddr_in);
        int *client = malloc(sizeof(int));
        *client = accept(sock, (struct sockaddr *)&source_addr, &addr_len);
        if (*client < 0) {
            ESP_LOGE(TAG, "Unable to accept connection: errno %d", errno);
        } else {
            ESP_LOGI(TAG, "Client " IPSTR ":%d connected", IP2STR((struct esp_ip4_addr *)&source_addr.sin_addr), source_addr.sin_port);
            xTaskCreate(tcp_client_task, "client_task", 2048, client, 6, NULL);
        }
    }

__exit:
    close(sock);
    free(ip_addr);
    vTaskDelete(NULL);
}

static void wifi_event_handler(void* arg,
                               esp_event_base_t event_base,
                               int32_t event_id,
                               void* event_data)
{
    if (event_base == IP_EVENT) {
        if (event_id == IP_EVENT_STA_GOT_IP) {
            ip_event_got_ip_t *data = event_data;
            esp_ip4_addr_t *ip_addr = malloc(sizeof(esp_ip4_addr_t));
            memcpy(ip_addr, &data->ip_info.ip, sizeof(esp_ip4_addr_t));
            xTaskCreate(tcp_server_task, "tcp_server", 2048, ip_addr, 5, &server_task_handle);
        }
    } else if (event_base == WIFI_EVENT) {
        if (event_id == WIFI_EVENT_STA_DISCONNECTED) {
            vTaskDelete(server_task_handle);
        } else if (event_id == WIFI_EVENT_STA_START) {
            esp_wifi_connect();
        }
    }
}

int app_main()
{
    /* 初始化NVS */
    esp_err_t ret = nvs_flash_init();
    if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) {
        ESP_ERROR_CHECK(nvs_flash_erase());
        ESP_ERROR_CHECK(nvs_flash_init());
    }

    /* 初始化WiFi协议栈 */
    ESP_ERROR_CHECK(esp_netif_init());
    ESP_ERROR_CHECK(esp_event_loop_create_default());
    esp_netif_create_default_wifi_sta();

    wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
    ESP_ERROR_CHECK(esp_wifi_init(&cfg));

    ESP_ERROR_CHECK(esp_event_handler_instance_register(WIFI_EVENT,
                                                        ESP_EVENT_ANY_ID,
                                                        &wifi_event_handler,
                                                        NULL,
                                                        NULL));

    ESP_ERROR_CHECK(esp_event_handler_instance_register(IP_EVENT,
                                                        ESP_EVENT_ANY_ID,
                                                        &wifi_event_handler,
                                                        NULL,
                                                        NULL));

    wifi_config_t wifi_config = {
        .sta = {
            .ssid = "Your SSID",
            .password = "Your password",
            .threshold.authmode = WIFI_AUTH_WPA_WPA2_PSK,
        },
    };

    ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA));
    ESP_ERROR_CHECK(esp_wifi_set_config(WIFI_IF_STA, &wifi_config));
    ESP_ERROR_CHECK(esp_wifi_start());

    return 0;
}

        服务器的代码与客户端是有一部分重合的,所以下面只重点介绍不同的地方。

1. 创建套接字

        参考上面。

2. 绑定IP与端口

        调用bind函数。传入的参数其实跟connect函数是一样的;但是这里的IP地址是自己的IP地址,端口的话就是自定义的。

3. 监听端口

        调用listen函数。第一个参数传入套接字描述符,第二个参数用来使能log记录。

4. 接受客户端连接

        调用accept函数。传入套接字描述符、IP地址结构体和结构体的长度。这个结构体是用来接收客户端的IP信息的,初始化为空即可。这个函数是阻塞的,只要没有客户端连接就不会返回;如果有客户端连接,就会返回该客户端的套接字描述符。

        例程中,一旦客户端成功连接就会创建一个线程处理这个客户端的数据,这样的话就能实现多客户端的连接服务。

5. 接收数据

        参考上面。如果recv函数返回0则代表客户端断开了连接,这时我们就可以关闭这个套接字,退出线程。

6. 发送数据

        参考上面。

        测试时先将ESP32上电,确保服务器已经启动并处于监听状态;然后在上位机这里设置为TCP客户端模式,填入ESP32的IP和端口;然后就可以连接并发送数据了。

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

搜索文章

Tags

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