最新资讯

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

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

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

 目录 

实现思路

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