最新资讯

  • 【Linux笔记】进程间通信——匿名管道||进程池

【Linux笔记】进程间通信——匿名管道||进程池

2025-04-27 06:00:29 0 阅读

🔥个人主页🔥:孤寂大仙V
🌈收录专栏🌈:Linux
🌹往期回顾🌹:【Linux笔记】动态库与静态库的理解与加载
🔖流水不争,争的是滔滔不


  • 一、Linux进程间通信简介和目的
  • 二、常见IPC方式
    • 管道
    • System V
  • 三、匿名管道
    • 工作原理
    • 特性
    • 管道读写规则
  • 四、基于匿名管道的进程池

一、Linux进程间通信简介和目的

简介

进程间通信(IPC,Inter-Process Communication)是操作系统中不同进程之间交换数据、协调工作的核心机制。由于每个进程拥有独立的内存空间,彼此隔离,因此需要特定的技术手段实现跨进程协作。

进程间通信(IPC)是指运行在同一台计算机或不同计算机上的多个进程之间进行数据交换和通信的技术。由于每个进程都有自己的地址空间,它们无法直接访问彼此的数据,因此需要通过特定的机制实现通信。IPC是操作系统和多进程编程中的一个重要概念,广泛应用于分布式系统、多任务操作系统以及各种应用程序之间。

进程间通信的本质:是先让不同的进程,先看到同一份资源,然后才有通信的条件。

目的

  • 数据传输:一个进程需要将它的数据发送给另一个进程
  • 资源共享:多个进程之间共享同样的资源。
  • 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止时要通知父进程)。
  • 进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变

二、常见IPC方式

管道

类型:

  • 匿名管道:用于父子进程间通信(半双工)。
  • 有名管道(FIFO):可用于不相关进程(全双工)。

特点:

  • 内核缓冲区存储数据,读写同步。
  • 数据读取后即删除,不支持随机访问。

System V

  • 共享内存(Shared Memory)

  • 核心机制:多个进程直接访问同一块物理内存。

  • 特点:
    速度最快,无需数据复制。
    需配合同步机制(如信号量、互斥锁)。

  • 消息队列(Message Queue)

  • 机制:进程通过发送 / 接收消息(带类型)通信。

  • 特点:
    消息按类型排序,支持异步通信。
    消息队列在内核中持久化,直到被显式删除。

  • 信号量(Semaphore)

  • 作用:控制对临界资源的访问,实现同步与互斥。

  • 类型:
    二元信号量:0/1 状态(互斥锁)。
    计数信号量:控制并发访问数量。

三、匿名管道

我们把从一个进程连接到另一个进程的一个数据流称为一个“管道”

工作原理


通过父子进程继承关系,再将文件描述符关闭,实现一端写,一端读,就是匿名管道。

匿名管道是单向通信(半双工)数据只能从管道的一端写入,另一端读取,无法双向传输。内核缓冲区存储数据通过内核临时存储,大小通常为 64KB(可通过 ulimit -a 查看)。生命周期短,管道随创建它的进程(父进程)结束而自动销毁,无需手动清理。亲缘关系限制仅适用于父子进程或兄弟进程(通过 fork/exec 系列函数创建的进程)。

父进程以读写方式打开文件。父进程fork创建子进程,(进程具有独立性)子进程要拷贝一份PCB结构,PCB中包含了files_struct结构,files_struct中有一个指向struct file(文件)的指针数组,而文件描述符就是这个数组的下标。在拷贝后,子进程也就有了指向struct file(文件)的对应数组元素下标(文件描述符)。而struct file(文件)是独属于文件的,和进程没有关系,也就不用拷贝,也就是说此时父子进程公共区域就是 struct file。(实现不同进程看到同一份资源)。write是系统调用接口,会将数据放在内核缓冲区,底层定期刷新缓冲区将内容写到磁盘。

  • 创建管道
    通过系统调用 pipe(int fd[2]) 创建匿名管道,返回两个文件描述符:
    fd[0]:读端(read end)。
    fd[1]:写端(write end)。

  • 数据流动
    父进程通过 write(fd[1], buf, len) 向管道写入数据。
    子进程通过 read(fd[0], buf, len) 读取数据。


子进程继承父进程的文件描述符表


匿名管道的示例

#include 
#include 
#include 
#include 
#include 
#include 

// 9:
void ChildWrite(int wfd)//子进程写
{
    char c = 0;
    int cnt = 0;
    char buffer[1024];
    while (true)
    {
        
        snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d
", getpid(), cnt++);
        write(wfd,buffer,strlen(buffer));
    }
}

void FatherRead(int rfd)//父进程读
{
    char buffer[1024];
    while (true)
    {
        // sleep(100);
        sleep(1);
        buffer[0] = 0;
        ssize_t n = read(rfd, buffer, sizeof(buffer)-1);
        if(n > 0)
        {
            buffer[n] = 0;
            std::cout << "child say: " << buffer << std::endl;
            // sleep(2);
        }
        else if(n == 0)
        {
            std::cout << "n : " << n << std::endl;
            std::cout << "child 退出,我也退出";
            break;
        }
        else
        {
            break;
        }

        break;
    }
}

int main()
{
    // 1. 创建管道
    int fds[2] = {0}; // fds[0]:读端   fds[1]: 写端
    int n = pipe(fds);
    if (n < 0)
    {
        std::cerr << "pipe error" << std::endl;
        return 1;
    }
    std::cout << "fds[0]: " << fds[0] << std::endl;
    std::cout << "fds[1]: " << fds[1] << std::endl;

    // 2. 创建子进程
    pid_t id = fork();
    if (id == 0)
    {
        // child
        // code
        // 3. 关闭不需要的读写端,形成通信信道
        // f -> r, c -> w
        close(fds[0]);
        ChildWrite(fds[1]);
        close(fds[1]);
        exit(0);
    }
    // 3. 关闭不需要的读写端,形成通信信道
    // f -> r, c -> w
    close(fds[1]);
    FatherRead(fds[0]);
    close(fds[0]);


    int status = 0;
    int ret = waitpid(id, &status, 0); // 获取到子进程的退出信息吗!!!
    if(ret > 0)
    {
        printf("exit code: %d, exit signal: %d
", (status>>8)&0xFF, status&0x7F);
    }
    return 0;
}

以上代码通过子进程写父进程写,形成信道。

特性

  • 匿名管道,只能用来进行具有血缘关系的进程进行进程间通信。
  • 管道是面向字节流的
  • 管道是单向通信的-需要半双工的一种特殊情况
  • 管道文件自带同步机制

同步机制的四种情况:

  • 写入慢,读取快。读端就要阻塞,等写端的操作。

读端等待写端进行操作,写一条读一条。

  • 写入快,读取慢。读端满了,写端就要阻塞。

把读取端也就是父进程,设置为slee(5),一瞬间写端子进程写满了,等待读端进程进行读取

  • 写关,读端继续读。read就会读到返回值为0,表示文件的结尾。
  • 读端关闭,写端继续写。写端的写入无意义。操作系统会杀掉进程。

管道读写规则

当没有数据可读时

  • O_NONBLOCK disable:read调用阻塞,即进程暂停执行,一直等到有数据来到为止。
  • O_NONBLOCK enable:read调用返回-1,errno值为EAGAIN。

当管道满的时候

  • O_NONBLOCK disable: write调用阻塞,直到有进程读走数据
  • O_NONBLOCK enable:调用返回-1,errno值为EAGAIN
    如果所有管道写端对应的文件描述符被关闭,则read返回0
    如果所有管道读端对应的文件描述符被关闭,则write操作会产生信号SIGPIPE,进而可能导致write进程

退出

当要写入的数据量不大于PIPE_BUF时,linux将保证写入的原子性。
当要写入的数据量大于PIPE_BUF时,linux将不再保证写入的原子性。

四、基于匿名管道的进程池

池化技术(Pooling Technology) 是一种通过预先创建并复用资源实例来优化系统性能的技术。其核心思想是减少资源频繁创建与销毁的开销,提升资源利用率。以下是关于池化技术的详细说明:

  • 预创建资源:在系统初始化时,预先创建一定数量的资源实例(如进程、线程、数据库连接等)。
  • 复用机制:当任务需要资源时,直接从池中获取已创建的实例,而非新建;任务完成后,资源归还池内供后续任务使用。
  • 动态管理:部分池化技术支持根据负载动态调整资源数量(如扩容或缩容)。

进程池(Process Pool)适用场景:CPU 密集型任务(如并行计算、批量数据处理)。
特点:每个进程独立拥有内存空间,避免全局变量冲突。适合多核 CPU,但内存占用较高。

以下是基于匿名管道实现的简单进程池

#ifndef _PROCESS_POOL_HPP_
#define _PROCESS_POOL_HPP_

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "Task.hpp"

using namespace std;

const int Poolnum = 5;
class Channel
{
public:
    Channel(int fd, pid_t pid)
        : _wfd(fd), _supid(pid)

    ~Channel() {}

private:

};

class ChannelManager
{
public:
    ChannelManager()
        : _next()
    {
    }

    ~ChannelManager() {}

private:
}

class ProcessPool
{
public:
    ProcessPool()
        : _process_num()
    {
    }
    void work(int rfd)
    {

    }
    bool Start()
    {
        for (int i = 0; i < _process_num; i++)
        {
            int pipefd[2] = {0}; // 创建管道
            int n = pipe(pipefd);
            if (n < 0)
            {
                return false;
            }
            pid_t supid = fork(); // 创建进程
            if (supid < 0)
            {
                return false;
            }
            else if (supid == 0)
            {
                // 子进程读
                close(pipefd[1]); // 关闭子进程写端
                work(pipefd[0]);
                close(pipefd[0]);
                exit(1);
            }
            else
            {
                // 父进程写
                close(pipefd[0]); // 关闭子进程读端
                _cm.Insert(pipefd[1], supid);
            }
        }
        return true;
    }

    ~ProcessPool() {}

private:
};
#endif

整体框架是,有一个channel类是对单个的信道进行描述的,一个channelmanager类是对多个信道进行描述,最后是processpool是对进程的读写以及整体进行描述的。


以下是procress这个类内部函数的理解

bool Start()
    {
        for (int i = 0; i < _process_num; i++)
        {
            int pipefd[2] = {0}; // 创建管道
            int n = pipe(pipefd);
            if (n < 0)
            {
                return false;
            }
            pid_t supid = fork(); // 创建进程
            if (supid < 0)
            {
                return false;
            }
            else if (supid == 0)
            {
                // 子进程读
                close(pipefd[1]); // 关闭子进程写端
                work(pipefd[0]);
                close(pipefd[0]);
                exit(1);
            }
            else
            {
                // 父进程写
                close(pipefd[0]); // 关闭子进程读端
                _cm.Insert(pipefd[1], supid);
            }
        }
        return true;
    }

这里创建进程和匿名管道,让子进程读父进程写。注意信道是多个的数量由你自己来设定,所以设置一个for循环进行创建多个子进程与信道。把父进程的写端和进程的pid传给具体的channel信道,具体的工作让具体的channel信道完成。

 ProcessPool(int num)
        : _process_num(num)
    {

    }
    void work(int rfd)
    {
        int code = 0;

        ssize_t n = read(rfd, &code, sizeof(code));
        if (n > 0)
        {
            if (n != sizeof(code))
            {
                return;
            }
            cout << "子进程pid->" << getpid() << "收到任务码->" << code << endl;
            _tm.Execute(code);
        }
        else if (n == 0)
        {
            cout << "子进程退出" << endl;
        }
        else
        {
            cout << "读取错误" << endl;
        }
    }

上面是构造和子进程的读端是如何工作的,在这里设置一个任务码code,让子进程根据任务码的不同去做不同的工作。

void Run()
    {
        int taskcode = _tm.Code(); // 选择一个任务
        auto &c = _cm.Select();    // 选择一个信道
        c.Send(taskcode);          // 发送任务码
    }
    void Stop()
    {
        // 父进程读端关闭
        _cm.StopSubProcess();
        // 回收子进程
        _cm.WaitSubProcess();
    }

上面代码是选择任务码,子进程需要收到任务码然后去做不同的工作,发给信道然后对应的子进程接收任务吗完成工作。这里在任务函数中在详细聊。 执行任务完成后父进程读端和子进程写端都需要关闭。


下面是channel类

class Channel
{
public:
    Channel(int fd, pid_t pid)
        : _wfd(fd), _supid(pid)
    {
        _name = "Channel-" + to_string(_wfd) + "-" + to_string(_supid);
    }

    void Send(int code)
    {
        int n = write(_wfd, &code, sizeof(code));
        (void)n;
    }
    void Close()
    {
        close(_wfd);
    }
    void Wait()
    {
        pid_t rid = waitpid(_supid, nullptr, 0);
        (void)rid;
    }
    string Name()
    {
        return _name;
    }

    ~Channel() {}

private:
    int _wfd;
    pid_t _supid;
    string _name;
};

构造接收写端和pid,send()函数对应processpool类中的发送任f务码,要在具体的channel信道中发送任务码给子进程,每个子进程才能接受到对应的任务码。这里的close和wait就是关闭写端和回收子进程,对应processpool中的void Stop()。


下面是ChannelManager类

class ChannelManager
{
public:
    ChannelManager(int next = 0)
        : _next(next)
    {
    }
    void Insert(int fd, int pid)
    {
        _channel.emplace_back(fd, pid);
    }

    Channel &Select()
    {
        auto &c = _channel[_next];
        _next++;
        _next %= _channel.size();
        return c;
    }

    void StopSubProcess()
    {
        for (auto &channel : _channel)
        {
            channel.Close();
            std::cout << "关闭: " << channel.Name() << std::endl;
        }
    }
    void WaitSubProcess()
    {
        for (auto &channel : _channel)
        {
            channel.Wait();
            std::cout << "回收: " << channel.Name() << std::endl;
        }
    }
    ~ChannelManager() {}

private:
    vector<Channel> _channel;
    int _next;
};

对通信信道用vector数组进程管理。 Channel &Select(),是选择任务码的时候在整个通信信道的数组中采用沦陷的方式进行选择信道。 void Insert(int fd, int pid)就是传过来写端和进程pid与channel的构造相关联与processpool的父进程传写端和pid相关联。void StopSubProcess() void WaitSubProcess()就是关闭写端和等待子进程与close()和wait()相关联。


任务码具体实现的函数

#pragma once
#include
#include
using namespace std;

typedef void(*task_t)();//函数指针

void PrintLog()
{
    cout<<"我是一个打印日志的任务"<<endl;
}
void DownLoad()
{
    cout<<"我是一个下载的任务"<<endl;
}
void UpLoad()
{
    cout<<"我是一个上传的任务"<<endl;
}


class TaskManager
{
public:
    TaskManager()
    {
        srand(time(nullptr)); //时间戳种子
    }

    int Code()//产生随机值
    {
        return rand()%_tasks.size();
    }

    void Register(task_t t)//注册任务
    {
        _tasks.push_back(t);
    }

    void Execute(int code)
    {
        if(code>=0 && code<_tasks.size())
        {
            _tasks[code]();
        }
    }
private:
    vector<task_t> _tasks;
};

用函数指针,根据时间戳产生随机值,然后把对应注册的任务放进存储任务的数组中,最后execute函数根据产生的任务码在储存任务的数组中选择任务。

子进程work就是调用execute()把收到的任务码给这个函数然后让这个函数进行选择对应的任务。

   void Run()
    {
        int taskcode = _tm.Code(); // 选择一个任务
        auto &c = _cm.Select();    // 选择一个信道
        c.Send(taskcode);          // 发送任务码
    }

在次重新聊一下processpool中的run函数,通过调用task.hpp的code函数,获得任务码,然后通过Select()函数选择以一个信道,然后发送任务码给子进程,子进程接受任务码然后子进程开始运行。


ProcessPool.hpp

#ifndef _PROCESS_POOL_HPP_
#define _PROCESS_POOL_HPP_

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "Task.hpp"
using namespace std;

const int Poolnum = 5;
class Channel
{
public:
    Channel(int fd, pid_t pid)
        : _wfd(fd), _supid(pid)
    {
        _name = "Channel-" + to_string(_wfd) + "-" + to_string(_supid);
    }

    void Send(int code)
    {
        int n = write(_wfd, &code, sizeof(code));
        (void)n;
    }
    void Close()
    {
        close(_wfd);
    }
    void Wait()
    {
        pid_t rid = waitpid(_supid, nullptr, 0);
        (void)rid;
    }
    string Name()
    {
        return _name;
    }

    ~Channel() {}

private:
    int _wfd;
    pid_t _supid;
    string _name;
};

class ChannelManager
{
public:
    ChannelManager(int next = 0)
        : _next(next)
    {
    }
    void Insert(int fd, int pid)
    {
        _channel.emplace_back(fd, pid);
    }

    Channel &Select()
    {
        auto &c = _channel[_next];
        _next++;
        _next %= _channel.size();
        return c;
    }

    void StopSubProcess()
    {
        for (auto &channel : _channel)
        {
            channel.Close();
            std::cout << "关闭: " << channel.Name() << std::endl;
        }
    }
    void WaitSubProcess()
    {
        for (auto &channel : _channel)
        {
            channel.Wait();
            std::cout << "回收: " << channel.Name() << std::endl;
        }
    }
    ~ChannelManager() {}

private:
    vector<Channel> _channel;
    int _next;
};

class ProcessPool
{
public:
    ProcessPool(int num)
        : _process_num(num)
    {
        _tm.Register(PrintLog);
        _tm.Register(DownLoad);
        _tm.Register(UpLoad);
    }
    void work(int rfd)
    {
        int code = 0;

        ssize_t n = read(rfd, &code, sizeof(code));
        if (n > 0)
        {
            if (n != sizeof(code))
            {
                return;
            }
            cout << "子进程pid->" << getpid() << "收到任务码->" << code << endl;
            _tm.Execute(code);
        }
        else if (n == 0)
        {
            cout << "子进程退出" << endl;
        }
        else
        {
            cout << "读取错误" << endl;
        }
    }
    bool Start()
    {
        for (int i = 0; i < _process_num; i++)
        {
            int pipefd[2] = {0}; // 创建管道
            int n = pipe(pipefd);
            if (n < 0)
            {
                return false;
            }
            pid_t supid = fork(); // 创建进程
            if (supid < 0)
            {
                return false;
            }
            else if (supid == 0)
            {
                // 子进程读
                close(pipefd[1]); // 关闭子进程写端
                work(pipefd[0]);
                close(pipefd[0]);
                exit(1);
            }
            else
            {
                // 父进程写
                close(pipefd[0]); // 关闭子进程读端
                _cm.Insert(pipefd[1], supid);
            }
        }
        return true;
    }
    void Run()
    {
        int taskcode = _tm.Code(); // 选择一个任务
        auto &c = _cm.Select();    // 选择一个信道
        c.Send(taskcode);          // 发送任务码
    }
    void Stop()
    {
        // 父进程读端关闭
        _cm.StopSubProcess();
        // 回收子进程
        _cm.WaitSubProcess();
    }
    ~ProcessPool() {}

private:
    int _process_num = 5;
    ChannelManager _cm;
    TaskManager _tm;
};
#endif

Task.hpp

#pragma once
#include
#include
using namespace std;

typedef void(*task_t)();//函数指针

void PrintLog()
{
    cout<<"我是一个打印日志的任务"<<endl;
}
void DownLoad()
{
    cout<<"我是一个下载的任务"<<endl;
}
void UpLoad()
{
    cout<<"我是一个上传的任务"<<endl;
}


class TaskManager
{
public:
    TaskManager()
    {
        srand(time(nullptr)); //时间戳种子
    }

    int Code()//产生随机值
    {
        return rand()%_tasks.size();
    }

    void Register(task_t t)//注册任务
    {
        _tasks.push_back(t);
    }

    void Execute(int code)
    {
        if(code>=0 && code<_tasks.size())
        {
            _tasks[code]();
        }
    }
private:
    vector<task_t> _tasks;
};

main.cc

#include "ProcessPool.hpp"
int main()
{
    ProcessPool pp(Poolnum);
    pp.Start();
    int ch=5;
    while(ch--)
    {
        pp.Run();
        sleep(2);
    }
    pp.Stop();
}

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

搜索文章

Tags

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