• 【Linux编程】TcpServer 类的设计与实现:构建高性能的 TCP 服务器(二)

【Linux编程】TcpServer 类的设计与实现:构建高性能的 TCP 服务器(二)

2025-04-26 17:34:40 1 阅读

TcpServer 类的设计与实现:构建高性能的 TCP 服务器

在现代网络编程中,构建一个高效、稳定的 TCP 服务器是许多网络应用的基础。本文将详细介绍一个基于 C++ 的 TcpServer 类的设计与实现,该类提供了创建 TCP 服务器、处理客户端连接、数据传输和接收等功能。通过这个类,我们可以更容易地理解和实现 TCP 通信的细节。

1. TcpServer 类概述

TcpServer 类是一个用于创建和管理 TCP 服务器的类。它封装了套接字创建、绑定、监听、客户端连接处理、数据发送和接收等操作,使得网络通信更加简洁和易于管理。

2. 类构造与析构
  • 构造函数 TcpServer::TcpServer(int _port)TcpServer::TcpServer(std::string _host, int _port) 初始化服务器的主机地址和端口,并创建套接字。
TcpServer::TcpServer(int _port) : TcpServer("0.0.0.0", _port) {
}

TcpServer::TcpServer(std::string _host, int _port) : host(_host), port(_port) {
    std::cout << "create tcp server start." << std::endl;
    socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (socket_fd == -1) {
        std::cout << "socket create error!";
        return;
    }
    int ret = set_epoll_mode(socket_fd, O_NONBLOCK);
    if (ret < 0) {
        std::cout << "epoll_mode failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    server_addr.sin_family = AF_INET;
    inet_pton(AF_INET, host.c_str(), &server_addr.sin_addr);
    server_addr.sin_port = htons(port);
    int opt = 1;
    setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    isCreate = true;
    std::cout << "create tcp server ok." << std::endl;
}
  • 析构函数 TcpServer::~TcpServer() 虚析构函数,确保派生类的析构函数被正确调用。
TcpServer::~TcpServer() {
}
3. 服务器启动与停止
  • 启动服务器 TcpServer::Start() 绑定套接字到指定端口,并开始监听。
void TcpServer::Start() {
    if (running || !isCreate) {
        std::cout << "TcpServer start failed!" << "running=" << running << ", port=" << port << std::endl;
        return;
    }
    auto ret = bind(socket_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));
    if (ret == -1) {
        std::cout << "bind faild:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    ret = listen(socket_fd, SOMAXCONN);
    if (ret == -1) {
        std::cout << "Listen failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    std::cout << "server open: " << host << ":" << port << std::endl;
    epoll_start();
}
  • 停止服务器 TcpServer::Stop() 关闭服务器并释放资源。
void TcpServer::Stop() {
    this->Close();
}
4. 资源管理
  • 关闭连接 TcpServer::Close() 关闭套接字和 epoll 文件描述符,释放资源。
void TcpServer::Close() {
    isCreate = false;
    running = false;
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_DEL, socket_fd, &socket_event);
    ret = close(socket_fd);
    std::cout << "socket_fd已关闭:" << ret << std::endl;
    for (TcpClient *client : clients)
        client->Close();
    ret = close(epoll_fd);
    std::cout << "TcpServer epoll_fd已关闭:" << ret << std::endl;
}
5. 非阻塞模式设置
  • 设置非阻塞模式 TcpServer::set_epoll_mode() 设置套接字为非阻塞模式。
int TcpServer::set_epoll_mode(int sock_fd, int mode) {
    int flags = fcntl(sock_fd, F_GETFL, 0);
    if (flags == -1) {
        std::cout << "epoll_mode failed:" << sock_fd << std::endl;
        return -1;
    }
    return fcntl(sock_fd, F_SETFL, flags | mode);
}
6. 客户端连接处理
  • 客户端接受线程 TcpServer::client_accept_thread() 处理客户端连接和数据事件。
void TcpServer::client_accept_thread() {
    struct epoll_event client_events[1024];
    while (running) {
        int ret = epoll_wait(epoll_fd, client_events, 1024, -1);
        if (ret < 0) {
            if (errno == EAGAIN || errno == EWOULDBLOCK)
                continue;
            else {
                std::cerr << "epoll_wait failed: " << ret << " : " << errno << " : " << strerror(errno) << std::endl;
                break;
            }
        }
        for (int n = 0; n < ret; ++n) {
            if (client_events[n].data.fd == socket_fd) {
                client_connect();
            } else {
                struct epoll_event client_event = client_events[n];
                auto client = std::find_if(clients.begin(), clients.end(), [&client_event](const TcpClient *_client)
                                           { return (_client->client_fd == client_event.data.fd); });
                if (client == clients.end())
                    continue;
                int ret = (*client)->data_receive(*client);
                if (ret == 0) {
                    clients.erase(client);
                    delete *client;
                }
            }
        }
    }
    std::cout << "服务已关闭,不再提供任何服务!" << std::endl;
    isDispose = true;
}
  • 客户端连接 TcpServer::client_connect() 接受客户端连接并添加到 epoll 监控。
void TcpServer::client_connect() {
    struct sockaddr_in client_addr;
    socklen_t addr_len = sizeof(client_addr);
    int client_fd = accept(socket_fd, (struct sockaddr *)&client_addr, &addr_len);
    std::cout << "accept:" << client_fd << " : " << strerror(errno) << std::endl;
    if (client_fd <= 0)
        return;

    TcpClient *client = new TcpClient;
    client->running = true;
    client->connected = true;
    client->client_fd = client_fd;
    client->local_addr = client_addr;
    int ret = set_epoll_mode(client_fd, O_NONBLOCK);
    if (ret == -1) {
        std::cout << "服务器接受客户端-set_epoll_mode failed:" << strerror(errno) << std::endl;
        client->Close();
        return;
    }
    client->add_epoll_event(client_fd, epoll_fd, EPOLLIN | EPOLLET);

    client->DataReceived = [this](void *sender, DataReceiveEventArgs *e)
    {
        DataReceived.Invoke(this, e);
    };

    client->recv_data = new char[client->recv_data_length];

    clients.push_back(client);
    std::cout << "新的客户端已接入:" << inet_ntoa(client_addr.sin_addr) << ":" << htons(client_addr.sin_port) << std::endl;
}
7. epoll 事件处理
  • 启动 epoll TcpServer::epoll_start() 创建 epoll 实例并添加监听套接字。
void TcpServer::epoll_start() {
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1) {
        std::cout << "poll_create1 failed:" << epoll_fd << std::endl;
        close(socket_fd);
        return;
    }

    socket_event.events = EPOLLIN;
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, socket_fd, &socket_event);
    if (ret == -1) {
        std::cout << "epoll_ctl failed:" << ret << std::endl;
        close(epoll_fd);
        return;
    }

    running = true;
    std::thread th = std::thread(&TcpServer::client_accept_thread, this);
    th.detach();
}

完整的代码:
TcpServer.h 头文件

#pragma once

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "DataReceiveEventArgs.h"
#include "TcpClient.h"

class TcpServer
{
public:
    TcpServer(int _port);
    TcpServer(std::string _host, int _port);
    ~TcpServer();

public:
    EventHandler<DataReceiveEventArgs> DataReceived;

public:
    void Start();
    void Stop();
    void Close();
    bool IsDispose();

private:
    void epoll_start();                        // epoll初始化(创建epoll)
    int set_epoll_mode(int sock_fd, int mode); // epoll模式--创建socket时,为非阻塞模式
    void client_accept_thread();
    void client_connect();

    sockaddr_in get_remote_addr(int sock);

private:
    int socket_fd;
    int epoll_fd;
    std::string host = "0.0.0.0";
    int port = 0;
    bool running = false;
    bool isCreate = false;
    int send_buff_size = 1024 * 1024;
    int recv_buff_size = 1024 * 1024;
    struct sockaddr_in server_addr;
    std::list<TcpClient *> clients;
    // 将监听套接字添加到 epoll 中,监控 EPOLLIN (表示有数据可读)事件
    struct epoll_event socket_event;
    bool isDispose = false;
};

TcpServer.cpp

#include "TcpServer.h"

TcpServer::TcpServer(int _port) : TcpServer("0.0.0.0", _port)
{
}

TcpServer::TcpServer(std::string _host, int _port) : host(_host), port(_port)
{
    std::cout << "create tcp server start." << std::endl;
    // AF_INET 表示使用 IPv4 协议
    // SOCK_STREAM 表示套接字的类型,表示 面向连接的流式套接字
    socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    // std::cout << "create tcp server socket_fd:" << socket_fd << std::endl;
    if (socket_fd == -1)
    {
        std::cout << "socket ceate error!";
        return;
    }
    // 文件描述符为非阻塞模式
    int ret = set_epoll_mode(socket_fd, O_NONBLOCK);
    if (ret < 0)
    {
        std::cout << "epoll_mode failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }

    server_addr.sin_family = AF_INET;

    // ip字符串转int
    inet_pton(AF_INET, host.c_str(), &server_addr.sin_addr);
    //server_addr.sin_addr.s_addr = ntohl(server_addr.sin_addr.s_addr);
    server_addr.sin_port = htons(port);

    // 以下设置表示当调用close关闭客户端时,立即释放端口,不等待
    // 在TCP服务端,客户端调用 close(client_fd) 关闭连接后,如果你尝试重新连接时出现端口没有完全释放的情况,通常是由于 TCP 连接的 TIME_WAIT 状态没有及时清理。这是 TCP 协议的正常行为。
    // 在TCP连接关闭后,端口会进入 TIME_WAIT 状态。这个状态的目的是确保最后的数据包能够正确到达。如果新的连接尝试在该端口上进行,而该端口仍然处于 TIME_WAIT 状态,就会出现端口被占用的情况。
    // TIME_WAIT 状态通常会持续一段时间(默认是4分钟,即240秒),这可以通过操作系统的内核参数来修改。

    // struct linger linger_opt;
    // linger_opt.l_onoff = 1;  // 启用
    // linger_opt.l_linger = 1; // 立即关闭
    // // 此方法亲测无效
    // // setsockopt(socket_fd, SOL_SOCKET, SO_LINGER, &linger_opt, sizeof(linger_opt));

    int opt = 1;
    setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    isCreate = true;
    std::cout << "create tcp server ok." << std::endl;
}

TcpServer::~TcpServer()
{
}

void TcpServer::Start()
{
    if (running || !isCreate)
    {
        std::cout << "TcpServer start failed!" << "running=" << running << ", port=" << port << std::endl;
        // close(socket_fd);
        return;
    }
    // 绑定套接字到指定端口
    auto ret = bind(socket_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));
    if (ret == -1)
    {
        std::cout << "bind faild:" << ret << std::endl;
        close(socket_fd);
        return;
    }

    // 开始监听
    ret = listen(socket_fd, SOMAXCONN);
    if (ret == -1)
    {
        std::cout << "Listen failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    std::cout << "server open: " << host << ":" << port << std::endl;

    epoll_start();
}

void TcpServer::Stop()
{
    this->Close();
}

void TcpServer::Close()
{
    isCreate = false;
    running = false;

    // 1. 删除socket_fd的epoll事件
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_DEL, socket_fd, &socket_event);

    // 2. 关闭监听套接字
    ret = close(socket_fd);
    std::cout << "socket_fd已关闭:" << ret << std::endl;

    // 3. 从 epoll 中移除所有连接的客户端文件描述符
    for (TcpClient *client : clients)
        client->Close();
    // 4. 关闭 epoll 文件描述符
    ret = close(epoll_fd);
    std::cout << "TcpServer epoll_fd已关闭:" << ret << std::endl;
}

bool TcpServer::IsDispose()
{
    return isDispose;
}

int TcpServer::set_epoll_mode(int sock_fd, int mode)
{
    /*
    O_NONBLOCK(非阻塞模式):如果设置了这个标志,表示该套接字(或文件)是非阻塞的,执行读写操作时不会阻塞调用进程或线程。
    套接字在没有数据可读或可写时不会让程序等待,而是立即返回。
    O_RDWR、O_WRONLY、O_RDONLY(访问模式):表示套接字的打开方式。
    O_APPEND(追加模式):指示文件或套接字在写操作时会追加数据。
    */
    int flags = fcntl(sock_fd, F_GETFL, 0); // 获取当前套接字的文件状态标志
    if (flags == -1)
    {
        std::cout << "epoll_mode failed:" << sock_fd << std::endl;
        return -1;
    }
    // 设置套接字为非阻塞模式
    return fcntl(sock_fd, F_SETFL, flags | mode);
}

void TcpServer::client_accept_thread()
{
    struct epoll_event client_events[1024];
    while (running)
    {
        // 阻塞等待事件
        int ret = epoll_wait(epoll_fd, client_events, 1024, -1);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EWOULDBLOCK)
                continue;
            else
            {
                std::cerr << "epoll_wait failed: " << ret << " : " << errno << " : " << strerror(errno) << std::endl;
                break;
            }
        }
        // 处理返回的事件
        for (int n = 0; n < ret; ++n)
        {
            if (client_events[n].data.fd == socket_fd)
            {
                // 如果是监听套接字的事件,有新的客户端连接
                client_connect();
            }
            else
            {
                struct epoll_event client_event = client_events[n];
                auto client = std::find_if(clients.begin(), clients.end(), [&client_event](const TcpClient *_client)
                                           { return (_client->client_fd == client_event.data.fd); });
                if (client == clients.end())
                    continue;
                // 客户端有数据
                int ret = (*client)->data_receive(*client);
                if (ret == 0)
                {
                    clients.erase(client);
                    delete *client;
                }
            }
        }
    }
    std::cout << "服务已关闭,不再提供任何服务!" << std::endl;
    isDispose = true;
}

void TcpServer::client_connect()
{
    struct sockaddr_in client_addr;
    socklen_t addr_len = sizeof(client_addr);
    int client_fd = accept(socket_fd, (struct sockaddr *)&client_addr, &addr_len); // 接受连接
    std::cout << "accept:" << client_fd << " : " << strerror(errno) << std::endl;
    if (client_fd <= 0)
        return;

    TcpClient *client = new TcpClient;
    client->running = true;
    client->connected = true;
    client->client_fd = client_fd;
    client->local_addr = client_addr;
    // 设置客户端套接字为非阻塞模式
    int ret = set_epoll_mode(client_fd, O_NONBLOCK);
    if (ret == -1)
    {
        std::cout << "服务器接受客户端-set_epoll_mode failed:" << strerror(errno) << std::endl;
        client->Close();
        return;
    }
    // client.SetSendBuffSize(send_buff_size);
    // client.SetRecvBuffSize(recv_buff_size);
    //  将新客户端套接字添加到 epoll 中,监听可读事件

    // client->create_epoll();
    client->add_epoll_event(client_fd, epoll_fd, EPOLLIN | EPOLLET);

    client->DataReceived = [this](void *sender, DataReceiveEventArgs *e)
    {
        DataReceived.Invoke(this, e);
    };

    // client->start_receive();
    client->recv_data = new char[client->recv_data_length];

    clients.push_back(client);
    std::cout << "新的客户端已接入:" << inet_ntoa(client_addr.sin_addr) << ":" << htons(client_addr.sin_port) << std::endl;
}

void TcpServer::epoll_start()
{
    // 创建epoll
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1)
    {
        std::cout << "poll_create1 failed:" << epoll_fd << std::endl;
        close(socket_fd);
        return;
    }

    // 监听可读事件
    socket_event.events = EPOLLIN;
    // 将监听套接字的文件描述符传给 epoll
    socket_event.data.fd = socket_fd;
    // 将监听套接字添加到 epoll 中,监控 EPOLLIN 事件(表示有数据可读)
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, socket_fd, &socket_event);
    if (ret == -1)
    {
        std::cout << "epoll_ctl failed:" << ret << std::endl;
        close(epoll_fd);
        return;
    }

    running = true;
    std::thread th = std::thread(&TcpServer::client_accept_thread, this);
    th.detach();
}

8. 总结

本文详细介绍了 TcpServer 类的设计与实现,包括构造与析构、服务器启动与停止、资源管理、非阻塞模式设置、客户端连接处理以及 epoll 事件处理。通过这个类,我们可以更容易地理解和实现 TCP 通信的细节。这个类提供了一个简洁的接口来管理 TCP 服务器,使得网络编程更加高效和易于维护。

TcpServer 类的设计注重模块化和可扩展性,允许开发者根据具体需求进行定制和扩展。通过使用 epoll 事件模型,TcpServer 类能够支持高并发的客户端连接,适用于需要处理大量并发连接的网络应用。此外,类中的非阻塞模式设置和资源管理机制确保了服务器的稳定性和高效性。

总的来说,TcpServer 类为构建高性能的 TCP 服务器提供了一个强大的基础。通过这个类,开发者可以快速构建和部署 TCP 服务器,满足各种网络应用的需求。

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

搜索文章

Tags

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