最新资讯

  • 【Linux】进程替换与自定义 Shell:原理与实战

【Linux】进程替换与自定义 Shell:原理与实战

2025-05-01 19:00:44 0 阅读

目录

一、进程程序替换

1、替换原理

2、替换函数

(1)函数解释

  ① filename / pathname

 ② 参数表传递

 ③ 环境变量表传递

(2)命名理解

 二、自定义shell命令行解释器

1、实现原理

2、实现代码

(1)获取环境变量

(2)输出命令行提示符

(3)获取用户输入的命令

(4)命令行分析

(5)检测并执行内键命令

(6)执行命令

(7)完整代码

 三、函数和进程之间的相似性


一、进程程序替换

fork()函数创建新的子进程后,子进程如果想执行一个全新的程序呢?进程的程序替换来完成这个功能!程序替换是通过特定的接口,加载磁盘上的一个全新程序(代码和程序),加载到调用进程的地址空间中!

1、替换原理

用 fork 创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种 exec 函数以执行另一个程序。当进程调用一种 exec 函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用 exec 并不创建新进程,所以调用 exec 前后该进程的ID并未改变。exec只是用磁盘上的一个新程序替换了当前进程的正文段、数据段、堆段和栈段。

示例:

#include
#include

int main()
{
    printf("程序运行!
");
    execl("/usr/bin/ls", "ls", "-l", "-a", NULL);
    printf("程序运行完毕!
");
    return 0;
}

$ ./proc # 原始代码中的第二个printf函数被替换,不执行!
程序运行!
total 32
drwxrwxr-x  2 zyt zyt  4096 Apr 18 17:13 .
drwxrwxr-x 16 zyt zyt  4096 Apr 18 17:07 ..
-rw-rw-r--  1 zyt zyt    58 Apr 18 17:12 Makefile
-rwxrwxr-x  1 zyt zyt 16000 Apr 18 17:13 proc
-rw-rw-r--  1 zyt zyt   190 Apr 18 17:13 test.c

一旦程序替换成功,就去执行新代码了,后续的原始代码也就不存在了(被覆盖了)。

 exec* 函数只有失败返回值,没有成功返回值。所以对于exec函数不用对返回值做判断。

2、替换函数

● 这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回。

● 如果调用出错则返回-1,若成功,不返回。

#include 
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ...,char *const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);
int fexecve(int fd, char *const argv[], char *const envp[]);

(1)函数解释

  ① filename / pathname

pathname就是取【路径名+程序】作为参数;filename就是文件名作为参数,当指定filename作为参数时:

● 如果filename中包含“/”,则将其视为路径名。

● 否则就按照PATH环境变量,在它所指定的各目录中搜索可执行文件。【不知道PATH环境变量的话看前文:深入浅出:环境变量与进程地址空间的核心机制-CSDN博客】

 ② 参数表传递

函数execl、execlp 和 execle 要求新程序的每一个命令行参数都说明一个单独的参数。这种参数表以空指针结尾。对execl、execlp 和 execle三个函数表示命令行参数的一般方法:

char *arg(), char *argl, ..., char *argn, (char *)0

 这种语法显示的说明了最后一个命令行参数之后跟了一个空指针。如果用常量0来表示一个空指针,则必须将它强制转化成为一个指针:否则它将被解释成整形参数。如果一个整形数的长度与char*的长度不同,那么exec函数的实际参数将出错。

对于其余的4个函数,则应先构造一个指向各参数的指针数组,然后将该数组指针地址作为这4个函数的参数(建立argv)。

 ③ 环境变量表传递

以e结尾的三个函数(execle、execve、fexecve)可以传递一个指向环境字符串指针数组的指针。在ISO C原型(国际标准化组织制定的C语言标准)之前,execle的参数是:

char *pathname, char *arg(), ..., char *argn, (char *)0, char *envp[]  

从中可见,最后一个参数是指向环境字符串的各字符指针构成的数组的指针。而在ISO C原型中,所有命令行参数、空指针和envp指针都是用省略号(...)表示。 

其他四个函数,则使用调用进程中的environ变量为新程序复制现有的环境。通常,一个进程允许将其环境传播给其子进程,但有时也有这种情况,进程想为子进程指定某一个确定的环境,可以直接用putenv()函数【添加或修改环境变量,它接受一个形式为 "name=value" 的字符串,并将其添加到当前进程的环境变量列表中】;或者通过第三方变量environ。

(2)命名理解

这7个函数的参数很难记忆。函数名的字符会给我们一些帮助。

● 字母p表示该函数取filename作为参数,并且用PATH环境变量寻找可执行文件。

● 字母l表示该函数取一个参数表,参数以可变参数列表的形式传递。 

字母v与字母l互斥,v表示该函数取了一个argv[]矢量,参数以字符指针数组的形式传递。

● 字母e表示该函数取envp[]数组,自己维护环境变量,而不使用当前环境。

调用示例如下:

#include 

int main() {
    char *const argv[] = {"ps", "-ef", NULL};
    char *const envp[] = {"PATH=/bin:/usr/bin", "TERM=console", NULL};

    // 使用 execl 执行 /bin/ps 命令,需要完整路径
    execl("/bin/ps", "ps", "-ef", NULL);

    // 使用 execlp 执行 ps 命令,带 p 的,可以使用环境变量 PATH,无需写全路径
    execlp("ps", "ps", "-ef", NULL);

    // 使用 execle 执行 ps 命令,带 e 的,需要自己组装环境变量
    execle("/bin/ps", "ps", "-ef", NULL, envp);

    // 使用 execv 执行 /bin/ps 命令,需要完整路径
    execv("/bin/ps", argv);

    // 使用 execvp 执行 ps 命令,带 p 的,可以使用环境变量 PATH,无需写全路径
    execvp("ps", argv);

    // 使用 execve 执行 /bin/ps 命令,带 e 的,需要自己组装环境变量
    execve("/bin/ps", argv, envp);

    exit(0);
}

在很多Unix的实现中,这7个函数中只有execve是内核的系统调用。另外6个只是库函数,它们最终都要调用该系统调用。这7个函数之间的关系如图:

 二、自定义shell命令行解释器

1、实现原理

考虑下面这个与shell的互动:

[root@localhost epoll]# ls
client.cpp  readme.md  server.cpp  utility.h

[root@localhost epoll]# ps
  PID TTY      TIME     CMD
 3451 pts/0    00:00:00 bash
 3514 pts/0    00:00:00 ps

用下图的时间轴来表示事件的发生次序。其中时间从左向右,shell由标识为sh的方块表示,它随着时间的流逝从左向右移动。shell从用户读入字符串“ls”。shell建立一个新的进程,然后在那个进程中运行ls程序并等待那个进程结束。  然后shell读取新的一行输入,建立一个新的进程,在这个进程中运行程序并等待这个进程结束。

所以,实现一个自定义shell需要循环以下过程:

(1)获取命令行

(2)解析命令行

(3)建立一个子进程(fork)

(4)替换子进程(execvp)

(5)父进程等待子进程退出(wait)

2、实现代码

(1)获取环境变量

要实现一个自定义 Shell,环境变量的处理是关键部分。正常情况shell启动要从系统中获取环境变量,而我们自定义shell的环境变量信息要从父shell中获取。

手动获取:在 C 中可以通过 extern char **environ 访问继承的环境变量。将父Shell的环境变量逐条复制到g_env中,然后将g_env中的变量添加到当前进程的环境变量表。【注意:(这里使用了putenv() 函数)此后g_env[i]的内存由系统管理,不能再手动free!Unix/Linux系统提供的全局变量,指向当前进程的环境变量表,】

// 环境变量表
#define MAX_ENVS 1024
char *g_env[MAX_ENVS];
int g_envs = 0;

// 从父shell1获取环境变量表,本来是要从配置文件获取的
void InitEnv()
{
    extern char **environ;
    memset(g_env, 0, sizeof(g_env)); // 初始化
    g_envs = 0;

    // 1.获取环境变量
    for(int i = 0; environ[i]; i++)
    {
        // (1)申请空间
        g_env[i] = (char*)malloc(strlen(environ[i])+1);
        // (2)拷贝到我们的环境变量表
        strcpy(g_env[i], environ[i]);
        g_envs++;
    }
    // 测试:添加了一个新的环境变量
    g_env[g_envs++] = (char*)"HAHA=for_test";
    g_env[g_envs] = NULL; // 表的最后一个成员必须是NULL

    // 2.获取到的环境变量导入shell的表中
    for(int i = 0; g_env[i]; i++)
    {
        putenv(g_env[i]);
    }
}

(2)输出命令行提示符

提示符格式:[用户名@主机名:当前目录的基名]# 

① MakeCommandline() 中使用 snprintf() 格式化提示符:【format:格式化字符串,用于指定如何格式化后续的参数】【...:可变参数列表,根据 format 的要求提供相应的值】

int snprintf(char *str, size_t size, const char *format, ...);

② PrintCommandline() 打印命令行提示符:fflush用来确保提示符立即显示(尤其在无换行符时)。

#define COMMAND_SIZE 1024
#define HOSTNAME_MAX 1024
#define FORMAT "[%s@%s:%s]# "

// shell定义的全局变量

char cwd[1019];
char cwdenv[1024]; 

const char *GetUserName()
{
    const char * name = getenv("USER");
    return name == NULL ? "None" : name;
}

const char *GetHostName()
{
    char hostname[HOSTNAME_MAX];
    return (gethostname(hostname, HOSTNAME_MAX) == -1) ? "None" : hostname;
}

const char *GetPwd()
{
    //const char * pwd = getenv("PWD");
    const char *pwd = getcwd(cwd, sizeof(cwd)); // 通过系统调用获取
    if(pwd != NULL)
    {   // 把自己的环境变量导给进程
        snprintf(cwdenv, sizeof(cwdenv), "PWD=%s", cwd);
        putenv(cwdenv);
    }
    return pwd == NULL ? "None" : pwd;
}

// 对路径做包装,只显示当前位置
std::string DirName(const char *pwd)
{
#define SLASH "/"
    std::string dir = pwd;
    if(dir == SLASH) return SLASH;
    auto pos = dir.rfind(SLASH);
    if(pos == std::string::npos) return "BUG";
    return dir.substr(pos+1);
}

// 制作命令行提示符
void MakeCommandline(char cmd_prompt[], int size)
{
    // 设置式化输入
    snprintf(cmd_prompt, size, FORMAT, GetUserName(), GetHostName(), DirName(GetPwd()).c_str());
    //snprintf(cmd_prompt, size, FORMAT, GetUserName(), GetHostName(), GetPwd());
}

// 打印命令行提示符
void PrintCommandline()
{
    char prompt[COMMAND_SIZE];
    MakeCommandline(prompt, COMMAND_SIZE);
    printf("%s", prompt);
    fflush(stdout);
}

(3)获取用户输入的命令

GetCommandline() 从标准输入读取一行用户输入的命令。

fgets() 是一个在 C 语言中用于从文件流中读取字符串的函数,属于标准库 中的函数。它能够从指定的文件流中读取一行数据,并将其存储到目标缓冲区中,同时可以限制读取的最大字符数,从而避免缓冲区溢出。

char* fgets(char* str, int size, FILE* stream);

但 fgets() 会保留输入中的换行符,所以调用后要清理换行符。

// 获取用户输入命令
bool GetCommandline(char *out, int size)
{
    // 从标准输入流获取命令,其实就是字符串
    char* c = fgets(out, size, stdin);

    if(c == NULL) return 1;
    out[strlen(out)-1] = ''; // 清理
, 至少会按一次回车,所以不会出错

    if(strlen(out) == 0) return false;// 什么都没输入
    return true;
}

(4)命令行分析

函数实现:命令行字符串的解析,将用户输入的命令行(如 "ls -a -l")拆分为 参数数组 g_argv,并记录参数个数 g_argc。

strtok() 是一个在 C 语言中用于字符串分割的函数,属于标准库 中的函数。它可以根据指定的分隔符将字符串分割成多个子字符串,并且在多次调用中逐个返回这些子字符串。

char* strtok(char* str, const char* delim);

在第一次调用时,str 是指向要分割的原始字符串的指针。 在后续调用中,str 应该传入 NULL,表示继续分割上一次调用后剩余的部分。

// 全局的命令行参数表
#define MAXARGC 128
char *g_argv[MAXARGC];
int g_argc = 0;

// 3. 命令行分析
bool CommandParse(char *commandline)
{
    // "ls -a -l" -> "ls" "-a" "-l"
    // 字符串切割 strtok()函数
#define SEP " "
    g_argc = 0;
    // 第一次调用:传入要分割的字符串和分隔符
    g_argv[g_argc++] = strtok(commandline, SEP);
    // 后续调用:传入 NULL 和分隔符,继续分割同一字符串
    while(g_argv[g_argc++] = strtok(NULL, SEP));
    g_argc--; // 修正参数计数,因为while 循环中 g_argc++ 在赋值后执行的
    return g_argc > 0 ? true : false;
}

(5)检测并执行内键命令

CheckAndExecBuiltin() 检查是否为内建命令。CD() 实现目录切换(cd 命令),Echo() 实现 echo 命令。

chdir() 是一个在 C 语言中用于更改当前工作目录的函数,属于标准库 中的函数(在 POSIX 系统中)。它允许程序动态地改变当前进程的工作目录。

// 最后一个程序退出码
int lastcode = 0;

// 全局的命令行参数表
#define MAXARGC 128
char *g_argv[MAXARGC];
int g_argc = 0;

// 内键命令
bool CD()
{

    if(g_argc == 1) // 只有一个cd时
    {
        std::string home = GetHome();
         if(home.empty()) return true;
        chdir(home.c_str()); // 切换路径
     }
    else
    {
        std::string where = g_argv[1]; // 目标路径
        if(where == "-")
        {}
        else if(where == "~")
        {}
        else
        {
            chdir(where.c_str());
        }
    }
    return true;
}

bool Echo()
{
    if(g_argc == 2)
    {
        //
        std::string opt = g_argv[1];
        if(opt == "$?") // eg1: echo $? 打印退出码
        {
            std::cout << lastcode << std::endl;
            lastcode = 0;
            return true;
        }
        if(opt[0] == '$') // eg2: echo $PATH 打印环境变量
        {
            std::string env_name = opt.substr(1);
            const char * env_value = getenv(env_name.c_str()); // 获取环境变量的值
            if(env_value) 
                std::cout << env_value << std::endl;
        }
    }
    return false;
}

// 4. 检测并执行内键命令
bool CheckAndExecBuiltin()
{
    std::string cmd = g_argv[0];
    if(cmd == "cd")
    {
       return CD();
    }
    else if(cmd == "echo")
    {
       return Echo();
    }
    return false;
}

(6)执行命令

Execute() 实现了 Shell 中外部命令的执行,核心是通过 fork() 创建子进程,并在子进程中调用 execvp 执行目标命令(自定义shell),父进程则等待子进程结束并记录其退出状态。

// 5.执行命令
int Execute()
{
    pid_t id = fork();
    if(id == 0)
    {
        // child, 程序替换
        execvp(g_argv[0], g_argv);
        exit(1);
    }
    // father
    int status = 0;
    pid_t rid = waitpid(id, &status, 0);
    if(rid > 0)
    {
        lastcode = WEXITSTATUS(status);
    }
    return 1;
}

(7)完整代码

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

#define COMMAND_SIZE 1024
#define HOSTNAME_MAX 1024
#define FORMAT "[%s@%s:%s]# "

// shell定义的全局变量

// 全局的命令行参数表
#define MAXARGC 128
char *g_argv[MAXARGC];
int g_argc = 0;

char cwd[1019];
char cwdenv[1024]; 

// 环境变量表
#define MAX_ENVS 1024
char *g_env[MAX_ENVS];
int g_envs = 0;

// 最后一个程序退出码
int lastcode = 0;

const char *GetUserName()
{
    const char * name = getenv("USER");
    return name == NULL ? "None" : name;
}

const char *GetHostName()
{
    char hostname[HOSTNAME_MAX];
    return (gethostname(hostname, HOSTNAME_MAX) == -1) ? "None" : hostname;
}

const char *GetPwd()
{
    //const char * pwd = getenv("PWD");
    const char *pwd = getcwd(cwd, sizeof(cwd)); // 通过系统调用获取
    if(pwd != NULL)
    {   // 把自己的环境变量导给进程
        snprintf(cwdenv, sizeof(cwdenv), "PWD=%s", cwd);
        putenv(cwdenv);
    }
    return pwd == NULL ? "None" : pwd;
}

const char *GetHome()
{
    const char *home = getenv("HOME");
    return home == "" ? NULL : home;
}

// 对路径做包装
std::string DirName(const char *pwd)
{
#define SLASH "/"
    std::string dir = pwd;
    if(dir == SLASH) return SLASH;
    auto pos = dir.rfind(SLASH);
    if(pos == std::string::npos) return "BUG";
    return dir.substr(pos+1);
}

// 制作命令行提示符
void MakeCommandline(char cmd_prompt[], int size)
{
    // 设置式化输入
    snprintf(cmd_prompt, size, FORMAT, GetUserName(), GetHostName(), DirName(GetPwd()).c_str());
    //snprintf(cmd_prompt, size, FORMAT, GetUserName(), GetHostName(), GetPwd());
}

// 打印命令行提示符
void PrintCommandline()
{
    char prompt[COMMAND_SIZE];
    MakeCommandline(prompt, COMMAND_SIZE);
    printf("%s", prompt);
    fflush(stdout);
}

// 获取用户输入命令
bool GetCommandline(char *out, int size)
{
    // 从标准输入流获取命令,其实就是字符串
    char* c = fgets(out, size, stdin);

    if(c == NULL) return 1;
    out[strlen(out)-1] = ''; // 清理
, 至少会按一次回车,所以不会出错

    if(strlen(out) == 0) return false;// 什么都没输入
    return true;
}

// 3. 命令行分析
bool CommandParse(char *commandline)
{
    // "ls -a -l" -> "ls" "-a" "-l"
    // 字符串切割 strtok()函数
#define SEP " "
    g_argc = 0;
    // 第一次调用:传入要分割的字符串和分隔符
    g_argv[g_argc++] = strtok(commandline, SEP);
    // 后续调用:传入 NULL 和分隔符,继续分割同一字符串
    while(g_argv[g_argc++] = strtok(NULL, SEP));
    g_argc--;
    return g_argc > 0 ? true : false;
}

void PrintArray()
{
    for(int i = 0; g_argv[i]; i++)
    {
        printf("argv[%d]->%s
", i, g_argv[i]);
    }
    printf("%d
",g_argc);
}

// 从父shell1获取环境变量表,本来是要从配置文件获取的
void InitEnv()
{
    extern char **environ;
    memset(g_env, 0, sizeof(g_env)); // 初始化
    g_envs = 0;

    // 1.获取环境变量
    for(int i = 0; environ[i]; i++)
    {
        // (1)申请空间
        g_env[i] = (char*)malloc(strlen(environ[i])+1);
        // (2)拷贝到我们的环境变量表
        strcpy(g_env[i], environ[i]);
        g_envs++;
    }
    // 测试:添加了一个新的环境变量
    g_env[g_envs++] = (char*)"HAHA=for_test";
    g_env[g_envs] = NULL; // 表的最后一个成员必须是NULL

    // 2.获取到的环境变量导入shell的表中
    for(int i = 0; g_env[i]; i++)
    {
        putenv(g_env[i]);
    }
}

// 内键命令
bool CD()
{

    if(g_argc == 1) // 只有一个cd时
    {
        std::string home = GetHome();
         if(home.empty()) return true;
        chdir(home.c_str());
     }
    else
    {
        std::string where = g_argv[1];
        if(where == "-")
        {}
        else if(where == "~")
        {}
        else
        {
            chdir(where.c_str());
        }
    }
    return true;
}

bool Echo()
{
    if(g_argc == 2)
    {
        //
        std::string opt = g_argv[1];
        if(opt == "$?") // eg1: echo $?
        {
            std::cout << lastcode << std::endl;
            lastcode = 0;
            return true;
        }
        if(opt[0] == '$') // eg2: echo $PATH
        {
            std::string env_name = opt.substr(1);
            const char * env_value = getenv(env_name.c_str()); // 获取环境变量的值
            if(env_value) 
                std::cout << env_value << std::endl;
        }
    }
    return false;
}

// 4. 检测并执行内键命令
bool CheckAndExecBuiltin()
{
    std::string cmd = g_argv[0];
    if(cmd == "cd")
    {
       return CD();
    }
    else if(cmd == "echo")
    {
       return Echo();
    }
    return false;
}

// 5.执行命令
int Execute()
{
    pid_t id = fork();
    if(id == 0)
    {
        // child, 程序替换
        execvp(g_argv[0], g_argv);
        exit(1);
    }
    // father
    int status = 0;
    pid_t rid = waitpid(id, &status, 0);
    if(rid > 0)
    {
        lastcode = WEXITSTATUS(status);
    }
    return 1;
}
// 释放通过malloc分配的环境变量内存
void FreeEnvMemory1()
{
    for(int i = 0; i < g_envs; i++) {
        if(g_env[i] != NULL) {
            free(g_env[i]);  // 释放每个环境变量字符串
            g_env[i] = NULL; // 置空指针
        }
    }
    g_envs = 0;  // 重置计数器
}

void FreeEnvMemory() {
    if (!g_used_putenv) {
        // 未调用 putenv,可以安全释放
        for (int i = 0; i < g_envs; i++) {
            free(g_env[i]);
            g_env[i] = NULL;
        }
    } else {
        // 调用过 putenv,只能释放数组指针(不释放字符串内存)
        for (int i = 0; i < g_envs; i++) {
            g_env[i] = NULL;  // 仅置空指针
        }
    }
    g_envs = 0;
}

int main()
{
    // shell启动要从系统中获取环境变量,而我们的环境变量信息要从父shell中获取
    InitEnv();
    // 当然是循环啦!
    while(true)
    {
        // 1、输出命令行提示符
        PrintCommandline();

        // 2. 获取输入的命令
        char commandline[COMMAND_SIZE];
        if(!GetCommandline(commandline, COMMAND_SIZE))
            continue;
        //printf("%s
", commandline);

        // 3. 命令行分析, 字符串拆分成多个元素
        if(!CommandParse(commandline)) continue;
        //PrintArray();

        // 4.检测并执行内建命令
        if(CheckAndExecBuiltin()) continue;

        // 5. 执行命令
        Execute();
    }
    // 释放空间
    FreeEnvMemory();
    return 0;
}

 三、函数和进程之间的相似性

一个C程序有很多函数组成。一个函数可以调用另一个函数,同时传递给它一些参数。被调用的函数执行一定的操作,然后返回一个值。每个函数都有他的局部变量,不同的函数通过 call/return 系统进行通信。这种通过参数和返回值在拥有私有数据的函数间通信的模式是结构化程序设计的基础。Linux 鼓励将这种应用于程序之内的模式扩展到程序之间。如下图:

一个C程序可以 fork/exec 另一个程序,并传给它一些参数。这个被调用的程序执行一定的操作,然后通过 exit(n) 来返回值。调用它的进程可以通过 wait(&ret) 来获取 exit 的返回值。

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

搜索文章

Tags

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