最新资讯

  • 【Linux】1w详解如何实现一个简单的shell

【Linux】1w详解如何实现一个简单的shell

2025-04-27 00:01:53 2 阅读

 目录 

实现思路

1. 交互 获取命令行

2. 子串分割 解析命令行

3. 指令的判断 内建命令

4. 普通命令的执行

 补充:vim 文本替换

整体代码

重点思考

1.getenv和putenv是什么意思

2.代码extern char **environ;

3.内建命令是什么

4.lastcode = WEXITSTATUS(status);

5.execvp(_argv[0], _argv);的调用

6._argc&_argv


实现思路

1. 交互 获取命令行

显示提示符和获取用户输入

Shell本质是一个死循环,不断地显示提示符和获取用户输入

memset 函数

memset 函数用于将一段内存区域设置为指定的值。它的原型是:

void *memset(void *s, int c, size_t n);

参数说明:

  • s:指向要填充的内存区域的指针。

  • c:要设置的值(以无符号字符形式传递,但实际存储在内存中的每个字节的值是该无符号字符的值)。

  • n:要设置的字节数。

示例用法:

char command_line[NUM];
memset(command_line, '', sizeof(command_line) * sizeof(char));

这里的代码表示将 command_line 数组的每个字节都设置为 (空字符),确保初始化整个数组。

fgets 函数

fgets 函数用于从指定的输入流读取字符串。它的原型是:

char *fgets(char *s, int n, FILE *stream);

参数说明:

  • s:指向存储读取数据的字符数组的指针。

  • n:要读取的最大字符数(包括终止字符 )。

  • stream:输入流,通常是 stdin 用于标准输入。

示例用法:

fgets(command_line, NUM, stdin);

这行代码表示从标准输入读取最多 NUM-1 个字符(预留一个字符用于终止字符 )到 command_line 数组中。

综合示例

结合起来,代码片段如下所示:

char command_line[NUM];
memset(command_line, '', sizeof(command_line) * sizeof(char));
fgets(command_line, NUM, stdin);

这段代码的作用是:

  1. 使用 memset 函数将 command_line 数组的所有字节都设置为 ,即初始化数组。

  2. 使用 fgets 函数从标准输入读取最多 NUM-1 个字符并存储在 command_line 数组中。

这样处理后,command_line 数组会包含从输入读取的字符串,并且如果字符串的长度小于 NUM,数组中剩余的字节会保持为 

以下是实现这两个步骤的代码:

#include 
#include 
#include 
#include 

#define NUM 1024

char command_line[NUM]; // 用来接收命令行内容

int main(void) {
    while (1) {
        /* Step1:显示提示符 */
        printf("[用户@主机 当前目录] # ");
        fflush(stdout);

        /* Step2:获取用户输入 */
        memset(command_line, '', sizeof(command_line));
        fgets(command_line, NUM, stdin);  // 从键盘获取输入
        command_line[strlen(command_line) - 1] = ''; // 消除 '
'
        printf("%s
", command_line);
    }
}

通过上述代码,我们可以实现提示用户输入,并获取用户输入。

注意点:

执行发现有空行怎么办

 我们利用 fgets 函数从键盘上获取,标准输入 stdin,获取到 C 风格的字符串,

注意默认会添加 ,我们先把获取到的结果 command_line 打印出来看看:

 因为 command_line 里有一个 ,我们把它替换成 即可:

 command_line[strlen(command_line) - 1] = '';  // 消除 ''

2. 子串分割 解析命令行

获取用户输入后,我们需要将接收到的字符串拆分为命令及其参数。

将接收到的字符串拆开

通过 strtok 函数,我们可以将一个字符串按照特定的分隔符打散,依次返回子串:

#include 
#include 
#include 
#include 

#define NUM 1024
#define SEP " "
#define SIZE 128

char command_line[NUM];
char* command_args[SIZE];

int main(void) {
    while (1) {
        /* 显示提示符和获取用户输入 */
        printf("[用户@主机 当前目录] # ");
        fflush(stdout);
        memset(command_line, '', sizeof(command_line));
        fgets(command_line, NUM, stdin);
        command_line[strlen(command_line) - 1] = '';

        /* 将接收到的字符串拆开 */
        command_args[0] = strtok(command_line, SEP);
        int idx = 1;
        while ((command_args[idx++] = strtok(NULL, SEP)));

        /* 打印拆分结果 */
        for (int i = 0; i < idx - 1; i++) {
            printf("%d : %s
", i, command_args[i]);
        }
    }
}

通过这段代码,我们可以将输入的命令行字符串拆分成多个子字符串,并打印出来。

strtok 函数的原型为:

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

参数说明:

  • str:要进行分割的字符串,第一次调用时传入要分割的字符串,后续调用时传入 NULL 即可。

  • delim:分隔符,用于指定分割字符串的字符。

示例用法

在代码中,使用了 strtok 函数将 command_line 字符串按照 SEP 分隔符进行切割,并将每个子字符串存储在 command_args 数组中。

command_args[0] = strtok(command_line, SEP);
int idx = 1;

这里的代码首先将 command_line 字符串按照 SEP 分隔符切割成子字符串,并将第一个子字符串的指针存储在 command_args[0] 中。然后,

利用循环逐个获取剩余的子字符串,并将它们存储在 command_args 数组中(使用 idx 来索引)

3. 指令的判断 内建命令

为了实现一些特定功能,如路径切换,我们需要在Shell中实现内建命令。

内建命令:实现路径切换

#include 
#include 
#include 
#include 

#define NUM 1024
#define SEP " "
#define SIZE 128

char command_line[NUM];
char* command_args[SIZE];

/* Shell 内置函数: 路径跳转 */
int ChangeDir(const char* new_path) {
    return chdir(new_path);
}

int main(void) {
    while (1) {
        /* 显示提示符和获取用户输入 */
        printf("[用户@主机 当前目录] # ");
        fflush(stdout);
        memset(command_line, '', sizeof(command_line));
        fgets(command_line, NUM, stdin);
        command_line[strlen(command_line) - 1] = '';

        /* 将接收到的字符串拆开 */
        command_args[0] = strtok(command_line, SEP);
        int idx = 1;
        while ((command_args[idx++] = strtok(NULL, SEP)));

        /* 判断并执行内建命令 */
        if (strcmp(command_args[0], "cd") == 0 && command_args[1] != NULL) {
            ChangeDir(command_args[1]);
            continue;
        }

        /* 执行普通命令 */
    }
}

这段代码通过判断输入的命令是否为 cd 来执行路径切换,而无需创建子进程。

getcwd用于获取当前工作目录(当前目录)的路径。该函数的声明如下:

char *getcwd(char *buf, size_t size);

函数参数说明:

  • buf:指向存储当前工作目录路径的缓冲区
  • size:缓冲区的大小

函数返回值: 如果函数调用成功,则返回指向存储当前工作目录路径的缓冲区的指针;如果函数调用失败,则返回NULL。

通过调用getcwd函数,可以获取当前程序所在的工作目录路径。


chdir用于改变当前工作目录(当前目录)的路径。该函数的声明如下:

int chdir(const char *path);

函数参数说明:

  • path:要设置为当前工作目录的路径

函数返回值: 如果函数调用成功,则返回0;如果函数调用失败,则返回-1,并设置errno来指示错误的类型。

4. 普通命令的执行

最后,我们实现普通命令的执行,包括创建子进程并执行用户输入的命令。

创建进程 & 程序替换

#include 
#include 
#include 
#include 
#include 
#include 
 
#define NUM 1024
#define SEP " "
#define SIZE 128
 
char command_line[NUM];
char* command_args[SIZE];
 
int main(void)
{
    while (1) {
        /* Step1:显示提示符 */
        printf("[lvy@我的主机名 当前目录] # ");
        fflush(stdout);
 
        /* Step2:获取用户输入 */
        memset (
            command_line, 
            '', 
            sizeof(command_line) * sizeof(char)
        );
        fgets(command_line, NUM, stdin);  /* 从键盘获取,标准输入,stdin 
            获取到 C 风格的字符串,默认添加 '' */
        command_line[strlen(command_line) - 1] = '';  // 消除 ''
 
        /* Step3: 将接收到的字符串拆开 - 字符串切分 */
        command_args[0] = strtok(command_line, SEP);
        int idx = 1;
        
        /* 这里的 = 是故意这么写的,因为 strtok 截取成功返回字符串起始地址
            截取失败,返回 NULL */
        while (command_args[idx++] = strtok(NULL, SEP));
 
        //我们来测试一下看看 
        // for (int i = 0; i < idx; i++) {
        //     printf("%d : %s
", i, command_args[i]);
        // }
 
        // printf("%s
", command_line);
 
        /* Step4. TODO */
        /* Step5. 创建进程,执行 */
        pid_t id = fork();
        if (id == 0) {
            /* child */
            /* Step6: 程序替换 */
            execvp (
                command_args[0],  // 保存的是我们要执行的程序名字
                command_args
            );
 
            exit(1);   // 只要执行到这里,子进程一定是替换失败了,直接退出。
        }
 
        /* Father */
        int status = 0;
        pid_t ret = waitpid(id, &status, 0);
        if (ret > 0) {   // 等待成功
            printf("等待成功!sig: %d, code: %d
", status&0x7F, (status>>8)&0xFF);
        }
    } // end while
 
}

通过上述代码,我们可以创建一个进程来执行用户输入的命令,并等待子进程结束。

给命令带颜色

为了增强Shell的用户体验,可以给一些常用命令添加颜色,例如 ls 命令:

/* 将接收到的字符串拆开 */
command_args[0] = strtok(command_line, SEP);
int idx = 1;
while ((command_args[idx++] = strtok(NULL, SEP)));

/* 给 ls 命令添加颜色 */
if (strcmp(command_args[0], "ls") == 0) {
    command_args[idx++] = (char*)"--color=auto";
}

以上实现了一个简单的Shell,具备了基本的提示符显示、用户输入获取、命令解析、内建命令和普通命令的执行功能。

内建命令 环境变量

/* Shell 内置函数: 路径跳转 */
int ChangeDir(const char* new_path) {
    chdir(new_path);
 
    return 0;  // 调用成功
}
 
int main(void) 
{
    ...
        /* Step4. TODO 编写后面的逻辑,内建命令 */
        if (strcmp(command_args[0], "cd") == 0 && command_args[1] != NULL) {
            ChangeDir(command_args[1]);  // 让调用方进行路径切换
            continue;
        }
    ...
}

保存环境变量的字符串,不能是易变的,所以 strcpy mycommand,实现与argv的分离

 补充:vim 文本替换

如何快速将mycmd换为myshell呢

通过如下操作

: %s/mycmd/myshell/g

就可以啦


细节设置的思考,在最后一部分,让我们先来看一下整体

整体代码

#include 
#include 
#include 
#include 
#include //创建子进程
#include //这些文件都是什么意思
#include 
#include 

#define LEFT "["
#define RIGHT "]"
#define LABLE "#"
#define DELIM " 	"
#define LINE_SIZE 1024
#define ARGC_SIZE 32
#define EXIT_CODE 44

int lastcode = 0;
int quit = 0;
extern char **environ;
char commandline[LINE_SIZE];
char *argv[ARGC_SIZE];//存储切割之后的命令行
char pwd[LINE_SIZE];

// 自定义环境变量表
char myenv[LINE_SIZE];
// 自定义本地变量表


const char *getusername()
{
    return getenv("USER");
}

const char *gethostname()
{
    return getenv("HOSTNAME");
}

void getpwd()
{
    getcwd(pwd, sizeof(pwd));//获取当前工作目录
}

void interact(char *cline, int size)//交互
{
    getpwd();
    printf(LEFT"%s@%s %s"RIGHT""LABLE" ", getusername(), gethostname(), pwd);
    char *s = fgets(cline, size, stdin);输入流进行输入
    assert(s);//断言不为空
    (void)s;//调用s避免报错
    // "abcd
"
    cline[strlen(cline)-1] = '';//取消自动换行
}

// ls -a -l | wc -l | head 
int splitstring(char cline[], char *_argv[])
{
    int i = 0;
    argv[i++] = strtok(cline, DELIM);//区分全局变量和形参_
    while(_argv[i++] = strtok(NULL, DELIM)); // 故意写的=
    //NULL的设置才能实现往后移的切割
    return i - 1;//去除NULL
}

void NormalExcute(char *_argv[])
{
    pid_t id = fork();
    if(id < 0){
        perror("fork");
        return;
    }
    else if(id == 0){
        //让子进程执行命令
        //execvpe(_argv[0], _argv, environ);
        execvp(_argv[0], _argv);
        //系统调用
        exit(EXIT_CODE);
    }
    else{
        int status = 0;
        pid_t rid = waitpid(id, &status, 0);
        if(rid == id) 
        {//返回正确执行
            lastcode = WEXITSTATUS(status);
        }
    }
}
//切换路径,内建命令
//shell执行的内建命令,一个一个判断
int buildCommand(char *_argv[], int _argc)
{
    if(_argc == 2 && strcmp(_argv[0], "cd") == 0){
        chdir(argv[1]);//切换路径的函数
        getpwd();//获取当前路径
        sprintf(getenv("PWD"), "%s", pwd);
        return 1;
    }
    //导入环境变量export
    else if(_argc == 2 && strcmp(_argv[0], "export") == 0){
        strcpy(myenv, _argv[1]);//为什么要进行一个拷贝 是什么意思呢!!!!
        putenv(myenv);//argv 是我们定义的,每次都是变化的
        //所以不要写我们的地址,要提字符串的地址
        return 1;
    }
    else if(_argc == 2 && strcmp(_argv[0], "echo") == 0){
        if(strcmp(_argv[1], "$?") == 0)
        {
            printf("%d
", lastcode);//查看退出码
            lastcode=0;
        }
        else if(*_argv[1] == '$'){//打印环境变量
            char *val = getenv(_argv[1]+1);//获取
            if(val) printf("%s
", val);
        }
        else{
            printf("%s
", _argv[1]);
        }

        return 1;
    }

    // 特殊处理一下ls
    if(strcmp(_argv[0], "ls") == 0)
    {
        _argv[_argc++] = "--color";
        _argv[_argc] = NULL;
    }
    return 0; //带有颜色之后返回,再执行普通命令
}

int main()
{
    while(!quit){
        //1.许多软件启动起来就是死循环
        // 2. 交互问题,获取命令行, ls -a -l > myfile / ls -a -l >> myfile / cat < file.txt
        interact(commandline, sizeof(commandline));
        //对函数做了一下封装

        // commandline -> "ls -a -l -n" -> "ls" "-a" "-l" "-n"
        // 3. 子串分割的问题,解析命令行
        int argc = splitstring(commandline, argv);
        //如何将字串打散呢
        //strtok需要循环调用
        //while(argv[i++]=strtok(commandline,DELIM);//故意写的等号
        if(argc == 0) continue;

        // 4. 指令的判断 
        // debug
        //for(int i = 0; argv[i]; i++) printf("[%d]: %s
", i, argv[i]);
        //内键命令,本质就是一个shell内部的一个函数
        int n = buildCommand(argv, argc);

        // ls -a -l | wc -l
        // 4.0 分析输入的命令行字符串,获取有多少个|, 命令打散多个子命令字符串
        // 4.1 malloc申请空间,pipe先申请多个管道
        // 4.2 循环创建多个子进程,每一个子进程的重定向情况。最开始. 输出重定向, 1->指定的一个管道的写端 
        // 中间:输入输出重定向, 0标准输入重定向到上一个管道的读端 1标准输出重定向到下一个管道的写端
        // 最后一个:输入重定向,将标准输入重定向到最后一个管道的读端
        // 4.3 分别让不同的子进程执行不同的命令--- exec* --- exec*不会影响该进程曾经打开的文件,不会影响预先设置好的管道重定向
        // 5. 普通命令的执行
        if(!n) NormalExcute(argv);//让命令0的时候执行
    }
    return 0;
}

重点思考

1.getenv和putenv是什么意思

 getenv函数用于获取指定环境变量的值。它的函数定义如下:

char *getenv(const char *name);
  • 参数:

    • name要获取的环境变量的名称。

  • 返回值:

    • 如果指定的环境变量存在,那么返回一个指向该环境变量值的指针

    • 如果指定的环境变量不存在,则返回NULL

以下是一个使用getenv函数的示例:

#include 
#include 

int main() {
    char *path = getenv("PATH");
    if (path != NULL) {
        printf("PATH environment variable: %s
", path);
    } else {
        printf("PATH environment variable not found.
");
    }
    return 0;
}

成功实现对环境变量的调用啦

putenv函数

putenv函数用于设置环境变量。它的函数定义如下:

int putenv(char *string);
  • 参数:

    • string:形式为"name=value"的字符串,用于设置具体的环境变量及其值

  • 返回值:

    • 成功时返回0。

    • 失败时返回非零值。

以下是一个使用putenv函数的示例:

#include 
#include 

int main() {
    char env_str[] = "MY_ENV=hello_world";
    if (putenv(env_str) == 0) {
        printf("Environment variable set successfully.
");
    } else {
        perror("putenv");
        return 1;
    }
    
    char *my_env = getenv("MY_ENV");
    if (my_env != NULL) {
        printf("MY_ENV: %s
", my_env);
    } else {
        printf("MY_ENV environment variable not found.
");
    }
    
    return 0;
}

注意事项

  1. 内存管理

    • getenv返回的指针指向的是环境变量的值,不能直接修改此值,否则可能导致未定义行为。

    • putenv函数参数所指向的字符串在函数调用后仍需存在,因为putenv不会复制这个字符串。因此传递给putenv的字符串应始终位于可修改的全局或堆内存中,而不是局部变量中。

  1. 线程安全性

    • getenvputenv函数在某些实现中不是线程安全的,特别是当修改同一个环境变量时。建议在多线程环境中使用setenvunsetenv函数,它们是现代C库中提供的线程安全的替代函数。

总结

getenvputenv是C语言中用于获取和设置环境变量的基本函数。通过了解并正确使用它们,可以更好地管理进程环境

2.代码extern char **environ;

 extern char **environ; 是C语言中的全局变量声明,用于访问当前进程的环境变量。为了理解这一行代码,我们需要理清以下几个关键概念:

环境变量的存储

在Unix和类Unix操作系统(如Linux)中,环境变量是一组键值对(例如PATH=/usr/bin),用于向进程传递配置信息。每个环境变量项以字符串的形式存储在一个全局变量数组中。这个数组在进程启动时由操作系统初始化,并且每个程序都可以访问和修改它。

环境变量在内存中的表示

在内存中,环境变量通常表示为一个字符串数组,每个字符串保存一个环境变量。例如:

PATH=/usr/bin
HOME=/home/user
USER=user
...

这些字符串指针存储在一个全局变量数组中,即char **environ

extern关键字

extern关键字用于声明一个全局变量,但不定义它。它告诉编译器这个变量是在别处(比如另一个源文件或由操作系统提供)定义的。因此,extern char **environ; 仅仅是一个声明,用来告知编译器这个变量在别处已经定义过,可以在当前文件中使用它。

为什么这样写?

在标准C库中,environ变量实际上在系统库中已经定义,我们只需要在我们的程序中声明一下即可使用。这种方式使我们能够访问和操作环境变量

这里是extern char **environ;的具体含义:

  1. 声明:它声明了一个外部变量environ,是一个指向字符指针的指针。

  2. 外部定义:实际的环境变量数组由操作系统初始化,并定义在某个系统库中。

  3. 全局访问:通过这个声明,我们可以在任何源文件中访问和操作环境变量。

示例

下面是一个具体的例子,展示了如何使用environ来访问并打印所有环境变量:

#include 

// 声明外部环境变量数组
extern char **environ;

int main(void) {
    // 指向环境变量数组的指针
    char **env = environ;

    // 遍历并打印所有环境变量
    while (*env) {
        printf("%s
", *env);
        env++;
    }

    return 0;
}

就可以成功调用所有环境变量啦

总结

extern char **environ; 这一行代码的作用是声明一个指针数组,用于访问当前进程的环境变量。通过这种方式,我们可以在C程序中方便地读取和操作环境变量

3.内建命令是什么

内建命令是指直接内置在操作系统内核中的一些命令,与普通的外部命令(外部程序文件)不同。这些内建命令是直接由shell解释器(如Bash、Zsh等)所处理,而不需要通过外部文件的方式来执行。这些内建命令通常在操作系统的shell环境中被频繁使用,并且执行速度更快,因为它们不需要创建新的进程来执行。

在Unix和类Unix操作系统中,通常会有一些内建命令,比如cdechoexit等。这些命令不需要单独的可执行文件,而是直接由shell内核提供支持。当用户在shell中输入这些命令时,shell会直接处理它们,而不需要通过搜索系统路径来找到可执行文件。

值得一提的是,某些shell也允许用户通过自定义的方式添加新的内建命令,这样用户可以根据自己的需求来扩展shell的内建功能。

4.lastcode = WEXITSTATUS(status);

在C语言中,WEXITSTATUS(status) 是一个宏,用于从waitwaitpid返回的状态信息中提取子进程的退出状态。这个宏主要用于处理子进程的退出状态信息

具体来说,WEXITSTATUS(status) 用于提取子进程在终止时传递给exit_exit函数的退出状态。这个宏将状态信息进行适当的位操作,以获取子进程的退出状态值。

一般情况下,status 是由waitwaitpid函数返回的子进程状态,其中包含了有关子进程终止的信息,包括退出状态。通过使用WEXITSTATUS(status),可以将状态转换为子进程的退出状态,以便于后续处理和判断子进程的终止情况。

具体的用法示例如下:

#include 
#include 
#include 
#include 
#include 

int main() {
    pid_t pid;
    int status;
    int lastcode;

    pid = fork();

    if (pid < 0) {
        perror("fork failed");
        exit(1);
    } else if (pid == 0) {
        // This is the child process
        char *args[] = {"ls", "-l", NULL};
        execvp(args[0], args);
    } else {
        // This is the parent process
        waitpid(pid, &status, 0);//获得了子进程的退出码
        lastcode = WEXITSTATUS(status);
        printf("子进程的退出状态是:%d
", lastcode);
    }

    return 0;
}

在这个例子中,WEXITSTATUS(status) 会从 status 中提取子进程的退出状态,并将其赋值给 lastcode。然后这个退出状态可以被用来进行一些处理,比如根据不同的退出状态进行不同的操作。

需要注意的是,使用 WEXITSTATUS(status) 的前提是要确保传入的 status 参数是一个子进程终止的状态,因为该宏只能提取终止进程的退出状态信息。

5.execvp(_argv[0], _argv);的调用

在代码中,execvp(_argv[0], _argv) 是一个执行函数 execvp 的调用,用于执行磁盘文件上的程序。这个函数会用指定的程序文件(由 _argv[0] 指定)来覆盖当前进程的镜像,并且用 _argv 数组中的参数替换掉原来的程序参数

相对路径执行指令

  1. 路径搜索根据 PATH 环境变量execvp 会在指定路径中查找可执行文件。
  2. 内存映射:找到可执行文件后,将其映射到当前进程地址空间。
  3. 替换镜像:用新程序的数据、堆栈、代码段替换当前进程的相应部分。
  4. 执行:新程序从其入口点开始执行,覆盖原进程的代码。

下面是对 execvp 函数调用的解释:

  • _argv[0] 表示要执行的程序文件的路径或名称。如果是一个程序的名称而没有路径,execvp 会在 $PATH 环境变量指定的路径中搜索这个程序。
  • _argv 是一个以空指针结尾的字符串数组,用于传递给新程序的命令行参数。数组的第一个元素(_argv[0])通常是被执行的程序的名称,随后的元素是程序的参数。
  • 当调用 execvp 时,操作系统会加载并执行指定的程序文件,并_argv 数组中的参数来替换当前进程的参数(因为默认会在PATH中查询,就和系统连接上了)
  • 如果 execvp 调用成功,则当前进程的镜像将被新程序替换,并且新程序开始执行。原来的程序代码和数据都会被新程序的代码和数据取代。
  • 如果 execvp 调用失败,它会返回-1,并且当前进程的状态不会改变。

在简单的C代码中,execvp 函数通常与 fork 函数一起使用,例如:

#include 
#include 
#include 

int main() {
    char *_argv[] = {"ls", "-l", "-a", NULL}; // 要执行的命令及参数组成的数组
    execvp(_argv[0], _argv);  // 在新的程序中执行 ls 命令
    // 如果执行成功,下面的代码不会被执行
    perror("execvp"); // 如果 execvp 失败,打印出错误信息
    return 1;
}

需要注意的是,execvp 在执行成功后,原进程的代码和数据将会被新进程替换。这就意味着,如果 execvp 后面还有代码,那么这些代码将不会被执行,因为当前的程序已经不再存在。

实现shell, 一行一行的运行,先判断是否为内建命令

6._argc&_argv

  • _argv:是一个字符指针数组,用于存储命令和参数。
  • _argc:是整型变量,用于存储命令和参数的数量
  • splitstring 函数将命令行字符串分割成多个子字符串,存储在 _argv 中,并返回子字符串的数量 _argc
  • NormalExcute 函数使用 _argv 数组创建子进程并执行命令。
  • buildCommand 函数使用 _argv 和 _argc 处理内建命令。

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

搜索文章

Tags

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