最新资讯

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

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

2025-04-30 00:00:17 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/5884.html

搜索文章

Tags

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