最新资讯

  • 【Linux】进程控制

【Linux】进程控制

2025-05-01 22:00:25 13 阅读

目录

  • 一、进程创建
    • 1.1 fork函数
    • 1.2 fork函数返回值
    • 1.3 写时拷贝
    • 1.4 fork常规用法
    • 1.5 fork调用失败的原因
    • 1.6 使用fork创建多进程
  • 二、进程退出
    • 2.1 进程退出场景
      • 2.1.1 进程运行完毕
      • 2.1.2 代码异常终止
      • 2.1.3 小结
    • 2.2 进程常见退出方法
      • 2.2.1 return
      • 2.2.2 调用exit函数
      • 2.2.3 调用_exit函数
      • 2.2.4 exit函数与_exit函数的区别
  • 三、进程等待
    • 3.1 什么是进程等待
    • 3.2 为什么要进行进程等待
    • 3.3 进程等待的方法
      • 3.3.1 wait函数
      • 3.3.2 waitpid函数
        • 3.3.2.1 参数pid
        • 3.3.2.2 参数status
        • 3.3.2.3 参数options
          • 3.3.2.3.1 参数option常用选项
          • 3.3.2.3.2 阻塞等待vs非阻塞等待
        • 3.3.2.4 返回值
      • 3.3.3 操作系统层面上父进程是如何获取子进程的退出信息
    • 3.4 父进程是在子进程的等待队列中等待的
    • 3.5 父进程等待多个子进程
  • 四、进程替换
    • 4.1 替换函数
      • 4.1.1 七种替换函数
      • 4.1.2 函数解释
      • 4.1.3 命名理解
    • 4.2 进程替换的使用
      • 4.2.1 单进程版程序替换的使用
      • 4.2.2 多进程版程序替换的使用
      • 4.2.3 各种替换函数在多进程的使用
        • 4.2.3.1 execl函数
        • 4.2.3.2 execlp函数
        • 4.2.3.3 execv函数
        • 4.2.3.4 execvp函数
        • 4.2.3.5 execle函数
        • 4.2.3.6 execvpe函数
      • 4.2.4 进程替换可以替换各种语言的进程
      • 4.2.5 进程替换中子进程获取环境变量
    • 4.3 进程替换的原理
      • 4.3.1 单进程替换的原理
      • 4.3.2 多进程替换的原理
      • 4.3.3 小知识点
      • 4.3.4 进程替换与程序加载到内存的关系
  • 结尾

一、进程创建

1.1 fork函数

我之前在进程的基本概念那篇文章中的进程创建中讲到过fork函数的原理,有兴趣的可以去那篇文章看看。进程的基本概念

在linux中fork函数时非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程

#include 
pid_t fork(void);
返回值:自进程中返回0,父进程返回子进程id,出错返回-1

进程调用fork,当控制转移到内核中的fork代码后,内核做:

  • 分配新的内存块和内核数据结构给子进程
  • 将父进程部分数据结构内容拷贝至子进程
  • 添加子进程到系统进程列表当中
  • fork返回,开始调度器调度



1.2 fork函数返回值

  • 子进程返回0
  • 父进程返回的是子进程的pid

1.3 写时拷贝

通常,父子代码共享,父子再不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自一份副本。具体见下图:

之前我们讲过当父进程创建完子进程后,子进程对数据进行写入时,会发生写时拷贝,这时操作系统会为子进程的这个数据重新申请空间,进行拷贝,修改页表的映射关系,但是此时子进程正在进行写入的操作,操作系统是怎么知道进行申请空间、进行拷贝和修改页表这些操作的时机的呢?

我们仔细观察上图发现父子进程的页表中所有数据的读写权限都是只读的,代码段中的数据是只读的我们可以理解,为什么数据段的数据也是只读的呢?将数据设置为只读就是为了解决操作系统如何找到进行申请空间、进行拷贝和修改页表这些操作的时机的,父进程在创建子进程时,会将自己页表中所有数据的读写权限都修改为只读,由于子进程的页表是拷贝父进程页表而来,那么父子进程的页表中所有数据的读写权限都是只读的,用户并不知道这些数据被修改为只读,他就会对某些数据进行写入,这时候页表转换就会出现权限问题,分为以下两种情况:

  1. 写入区域本就是只读的区域,这里就是真的出错了
  2. 写入区域原本是具有读写权限的区域,因为创建子进程被修改为只读权限,这里并不是出错了,而是触发我们重新申请空间、拷贝内容的策略机制,操作系统会在这时进行介入,进行申请空间和拷贝内容的操作,再将这个数据分别在对应页表的读写权限修改为读写

这里写实拷贝或许大家还有些疑问,前面的申请空间、修改页表、修改页表权限可能都没问题,但是为什么要进行拷贝呢?反正你都要进行写入,申请一个空间不就可以了吗?

其实并不是申请一个空间就行了的,你修改的可能是一整段数据的一部分,所以需要将原数据拷贝下来再进行局部修改,还有可能你是需要在原数据上进行修改,例如你要对原数据进行++操作。


1.4 fork常规用法

  • 一个父进程希望复制自己,使父子进程同时执行不同的代码段。例如,父进程等待客户端请求,生成子进程来处理请求。
  • 一个进程要执行一个不同的程序。

1.5 fork调用失败的原因

  • 系统中有太多的进程
  • 实际用户的进程数超过了限制

1.6 使用fork创建多进程

下面设计了一个程序,这个程序在运行起来的时候能够创建多个子进程,这些子进程能够与父进程做不同的任务,并且在子进程完成任务后直接退出,而父进程执行完任务以后休眠一段时间,使用一个监控脚本来查看进程的情况。

通过下图我们可以看到,父进程创建了三个子进程,子进程在完成了自己的任务以后都进入了僵尸状态,这个程序让所有的子进程都做了相同的事,后面可以优化使子进程分别做不同的事。

#include
#include
#include
    
#define N 3   
    
void Worker()    
{    
    int cnt = 3;    
    
    while(cnt--)    
    {    
        printf("I am a process , pid = %d , ppid = %d , cnt = %d
",getpid(),getppid(),cnt);    
        sleep(1);    
    }    
    
}    
    
int main()    
{    
    for(int i = 0; i < N ; i++)    
    {    
        sleep(1);    
                                                                                                                                            
        int id = fork();    
        if(id == 0)    
        {    
            printf("Create Child Process Success
");    
            Worker();    
            exit(0);    
        }    
    }

    sleep(100);

    return 0;
}

这里的代码是对上面代码的优化,将mian函数中的代码提取出来,单独变为一个创建子进程的函数(CreateSubProcess),我们在程序的开头定义了一个函数指针,函数可以作为CreateSubProcess函数的参数,CreateSubProcess函数可以通过参数控制创建几个子进程并让子进程干什么,接下来就只用写不同的函数,传给创建CreateSubProcess函数中,就能使这些子进程分别做不同的事。

#include    
#include    
#include    
    
typedef void(*callback)();    
    
#define N 3    
    
void Worker()    
{    
    int cnt = 3;                                                                                                                            
    
    while(cnt--)    
    {    
        printf("I am a process , pid = %d , ppid = %d , cnt = %d
",getpid(),getppid(),cnt);    
        sleep(1);    
    }    
    
}    
    
void CreateSubProcess(int n,callback cb)    
{    
    for(int i = 0; i < n ; i++)    
    {    
        sleep(1);    
    
        int id = fork();    
        if(id == 0)    
        {    
            printf("Create Child Process Success
");    
            cb();    
			exit(0);
        }
    }
}

int main()
{
    CreateSubProcess(N,Worker);
    sleep(100);

    return 0;
}

二、进程退出

2.1 进程退出场景

进程退出有以下三种场景:

  • 代码运行完毕,结果正确
  • 代码运行完毕,结果不正确
  • 代码异常终止

我们在写C语言代码的时候,每次在main函数结尾处都要写一个return 0,main函数也是函数,是函数就要被调用,那么这个0是返回给谁的呢?return的目的是什么?可以return其他值吗?

当我们运行一个进程时,进程的结尾一定是return,这个return一定需要用来表示进程的运行结果,return的值最终是返回给了该进程的父进程,return返回的值叫做退出码,我们可以使用echo $?来查看最近一个进程的退出码,'?'中保存的就是最近一次进程执行完毕后的退出码,通过下图的演示我们可以发现return不仅仅可以返回0还可以返回其他值。

接下来将在进程退出的三种场景进行讲解,这三个场景分为代码运行完毕和代码异常终止两种情况进行讲解。


2.1.1 进程运行完毕

代码运行完毕,分为结果正确和结果不正确两种情况,我们怎么知道结果正不正确呢?这里的我们指的是谁?在多进程的环境中,我们创建子进程的目的又是什么?

我们创建子进程的目的就是让子进程帮我们办事,这里的我们指的是父进程,父进程是如何知道子进程将事情办的怎么样的呢?就是通过上面讲述到的退出码,我们可以将退出码分为以下两种情况:

  1. 退出码为0,代表success,表示代码运行完毕,且进程运行结果正确
  2. 退出码为!0,代表failed,表示代码运行正常,但进程运行结果不正确,进程在运行时出现了错误,出现错误我们就需要知道错误的原因,退出码可以为1、2、3、4…,这些不同的数字可以表示不同的原因,纯数字可以表示出错原因,但是不方便人阅读,C语言内置的strerror函数能够将退出码转化为退出原因。

我们使用一些错误的指令来看看退出码和退出信息是否和上面一样对应,通过下图我们可以发现,有的退出码和退出信息与上面对应,而有的却不对应,这是因为如果我们不想使用C语言内置的退出码和退出信息对应关系,我们可以自定义退出码和退出信息对应关系。

这里对上面这部分内容进行总结:
main函数的退出码可以被父进程获取到,并且这个退出码可以用来判断子进程的运行结果。


在C语言的库中定义了一个全局变量错误码error,当一个库函数或系统调用失败时,会将error自动设置,当有多个函数调用失败时,error只会记录最后一个失败函数设置的值。

退出码vs错误码

  • 退出码通常是一个进程退出时的结果
  • 错误码通常衡量一个库函数或系统调用的调用情况

它们俩的共性就是能够在进程/函数出错时找到出错的具体原因

最常用退出码和错误码的用法就是在函数出错的时候将退出码和错误码保持一致,这就是使用系统默认退出码解决方案,如果你不想这样使用,就可以在函数出问题的时候就将错误信息打印出来。


2.1.2 代码异常终止

代码异常终止的本质就是进程收到了对应的信号,操作系统将这个进程终止了。


下图就是Linux操作系统中的信号大全,上图出现的两个错误分别对应的下面的8号信号和11号信号。

上面提到了代码异常终止的本质就是进程收到了对应的信号,操作系统将这个进程终止了,那我们向一个正常的进程发送信号,能不能将这个进程终止呢?通过下图代码的测试我们发现,对一个正常的进程发送信号能够让操作系统以对应的方式将进程终止。


2.1.3 小结

一个进程是否出异常,我们只要看有没有收到信号即可
一个进程结果是否正常,我们只要看返回码即可。


2.2 进程常见退出方法

正常终止(可以通过 echo $? 查看进程退出码):

2.2.1 return

对下面这段代码进行测试,return在main函数和在其他函数中的区别

  1. 在main中return代表进程结束,并将进程的退出码设置为这个值
  2. 在其他函数中return代表这个函数结束,这个值是作为函数的返回值
#include     
#include     
    
int func()    
{    
    printf("call func
");    
    
    return 10;    
}    
    
int main()    
{    
    func();    
    
    printf("I am a process , pid: %d , ppid: %d
",getpid(),getppid());    
    
    return 20;                                                                                                               
} 


2.2.2 调用exit函数

void exit(int status);

exit函数是C语言的库函数,exit函数的参数是进程退出码,在任意地方调用exit函数代表进程退出,不再进行后序的任何操作,对下面的代码进行测试,我们也发现func函数中调用exit函数,在main函数的输出语句确实没有调用。

#include                                                                                                            
#include     
#include     
    
void func()    
{    
    printf("call func
");    
    
    exit(10);    
}    
    
int main()    
{    
    func();    
    
    printf("I am a process , pid: %d , ppid: %d
",getpid(),getppid());    
    
    exit(20);    
}    


2.2.3 调用_exit函数

void _exit(int status);

_exit函数是系统调用,_exit函数的参数是进程退出码,在任意地方调用_exit函数代表进程退出,不再进行后序的任何操作,对下面的代码进行测试,我们也发现func函数中调用_exit函数,在main函数的输出语句确实没有调用。

#include     
#include     
#include     
    
void func()    
{    
    printf("call func
");    
        
    _exit(10);    
}    
    
int main()    
{    
    func();    
    
    printf("I am a process , pid: %d , ppid: %d
",getpid(),getppid());    
    
    _exit(20);                                                                                                               
}


2.2.4 exit函数与_exit函数的区别

下面这张图片是分别测试两个代码得到的,第一个代码和第二个代码的区别就是输出语句中是否带有’ ',再调用exit函数查看现象,我们可以发现两个程序都输出了,但是第一个代码是程序一运行就打印出字符串,而第二个代码是程序结束后才打印出字符串。

#include                                                                                                            
#include       
#include       
      
int main()      
{      
    printf("I am a code
");      
      
    sleep(3);    
                                                                     
    exit(0);                                                         
} 
#include                                                                                                            
#include       
#include       
      
int main()      
{      
    printf("I am a code");      
      
    sleep(3);    
                                                                     
    exit(0);                                                         
} 

下面这张图片是分别测试两个代码得到的,第一个代码和第二个代码的区别就是输出语句中是否带有’ ',再调用_exit函数查看现象,我们可以发现第一个代码是程序一运行就打印出字符串,而第二个代码是程序结束后也没有打印出字符串。

#include                                                                                                            
#include       
#include       
      
int main()      
{      
    printf("I am a code
");      
      
    sleep(3);    
                                                                     
    _exit(0);                                                         
} 
#include                                                                                                            
#include       
#include       
      
int main()      
{      
    printf("I am a code");      
      
    sleep(3);    
                                                                     
    _exit(0);                                                         
} 

小结
exit函数是库函数,_exit函数是系统调用,并且通过上面代码的测试我们发现,exit函数能够在进程结束后强制刷新缓冲区,而_exit函数在进程结束后不能够刷新缓冲区,exit函数本质上在底层封装的就是_exit函数,目前我们能够得到的结论就是缓冲区不在操作系统的内部。


三、进程等待

3.1 什么是进程等待

通过wait/waitpid的方式,让父进程对子进程进行资源回收的等待过程。


3.2 为什么要进行进程等待

  1. 解决子进程僵尸问题带来的内存泄漏问题。
  2. 父进程创建子进程是为了让子进程完成某些任务,父进程需要知道子进程任务完成的怎么样,所以父进程需要通过进程等待的方式来得到进程退出的信息。进程的退出信息就包括上面进程退出的三种情况,本质上就是获得子进程的信号编号和进程退出码。父进程有时候不需要子进程的退出信息,所以这两个数字并不是必须的,但是系统必须提供这样的基础功能,以防父进程需要。

3.3 进程等待的方法

3.3.1 wait函数

pid_t wait(int *status);

wait函数的参数,在waitpid函数统一讲解

wait函数能够等待任意子进程,wait函数的返回值:

  • 当wait函数等待成功后,返回子进程的pid
  • 当wait函数等待失败后,返回-1

通过对wait函数的使用可以得到以下结论:

  1. 父进程能够等待回收子进程僵尸状态,子进程的状态Z->X

通过对下面代码的运行和使用脚本观察进程可以得到下图,我们发现脚本中子进程先变为了僵尸进程,再消失了。

#include 
#include 
#include 
#include 
#include 

void Worker()
{
    int cnt = 3;
    while(cnt)
    {
        printf("I am child process , pid: %d , ppid : %d , cnt : %d
" ,getpid(),getppid(),cnt--);
        sleep(1);
    }      
}          
           
int main() 
{          
    pid_t id = fork();
           
    if(id == 0)
    {      
        // 子进程
        Worker();
                                                                                                                                            
        exit(0);
    }      
    else 
    {      
        // 父进程
        sleep(5);  // 用来观察子进程的僵尸状态

        pid_t rid = wait(NULL);
        if(rid == id)
            printf("Wait success , pid : %d
",getpid());

        sleep(3);
    }
    return 0;
}

  1. 如果子进程没有退出的情况下,父进程调用了wait函数,那么父进程就必须进行阻塞等待,直到子进程变为僵尸进程后,wait函数自动对子进程进行回收。

    我们之前讲阻塞状态是讲到过scanf函数,当进程调用到scanf函数时,我们不进行输入,键盘资源就没有就绪,进程就会变为阻塞状态。进程不仅仅可以等待硬件资源,还可以等待软件资源,等待进程也是等待资源,所以这时候父进程就会变为阻塞状态,当子进程运行完毕后,父进程的软件资源也就准备就绪了。

#include                                                                                             
#include     
#include     
#include                                                                                    
#include     
         
void Worker()    
{    
    int cnt = 3;    
    while(cnt)    
    {                     
        printf("I am child process , pid: %d , ppid : %d , cnt : %d
" ,getpid(),getppid(),cnt--);    
        sleep(1);    
    }    
}                    
                     
int main()    
{                   
    pid_t id = fork();    
            
    if(id == 0)    
    {                
        // 子进程                   
        Worker();    
    
        exit(0);    
    }                              
    else    
    {                    
        // 父进程                                            
        printf("Wait before
"); 
        
        pid_t rid = wait(NULL);

        printf("Wait after
");

        if(rid == id)
            printf("Wait success , pid : %d
",getpid());

        sleep(3);
    }
    return 0;
}

通过上面两个代码的运行我们可以得到一个结论:
一般而言,父子进程谁先运行我们不知道,但是最后一般都是父进程最后退出。


3.3.2 waitpid函数

waitpid函数在功能上可以完全替代wait函数,在后面的内容中会讲到。

pid_t waitpid(pid_t pid, int *status, int options);

3.3.2.1 参数pid

wait函数参数pid有常用的两个选项

  1. 指定一个子进程的pid,代表等待指定的子进程
  2. -1,代表等待任意子进程

3.3.2.2 参数status
  • wait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充。
  • 如果传递NULL,表示不关心子进程的退出状态信息。
  • 否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程,我们可以根据该参数获取到子进程的信号编号和退出码。
  • status不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status低16比特位):
#include     
#include     
#include     
#include     
#include     
    
void Worker()    
{    
    int cnt = 3;    
    while(cnt)    
    {    
        printf("I am child process , pid: %d , ppid : %d , cnt : %d
" ,getpid(),getppid(),cnt--);    
        sleep(1);    
    }    
}    
    
int main()    
{    
    pid_t id = fork();    
    
    if(id == 0)    
    {    
        // 子进程    
        Worker();    
    
        exit(10);                                                                                             
    }    
    else    
    {    
        // 父进程    
        printf("Wait before
"); 
        
        int status = 0;

        pid_t rid = waitpid(-1,&status,0);

        printf("Wait after
");

        if(rid == id)
            printf("Wait success , pid : %d , status : %d , exit sig : %d , exit code : %d
",getpid(),status , status>>8 , status&(0xFF));

        sleep(3);
    }
    return 0;
}

当我们在进程中故意写一个错误,观察信号编号和退出码,此时我们发现无论子进程的退出码是多少,只要子进程出现异常,那么子进程的退出码就是0,并且父进程可以得到子进程的信号编号。

上面是代码中出现错误导致子进程发现异常,若是子进程正常运行,我们给子进程发送信号会发生什么呢?我们发现同样可以使子进程终止,父进程也可以获得信号编号。

根据上面讲述的内容提三个问题:

  1. 当一个进程异常(收到信号),那么这个进程的退出码还有意义吗?
    答:没有任何意义。
  2. 我们怎么判断一个子进程有没有收到信号?
    答:信号编号全是大于0的,若信号编号为0则没有收到信号,反之则收到了信号。
  3. 我们为什么不定义一个全局变量status去获取子进程的退出信息?而使用系统调用去获得?
    答:因为进程具有独立性,当在子进程中修改status时,操作系统会对status进行写时拷贝,我们修改的是子进程中的status,而非父进程中的status,所以通过全局变量是无法让父进程获取子进程的退出信息的,需要系统调用来获取。

上面编写的代码中想获取子进程的信号编号和退出码还需要进行位操作,为了给不熟悉编程人员提供遍历,操作系统提供了下面两个函数

WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)


3.3.2.3 参数options
3.3.2.3.1 参数option常用选项

参数option有常用的两种选项:

  1. 0,父进程以阻塞等待的方式进行等待
  2. WNOHANG,父进程以非阻塞方式等待

3.3.2.3.2 阻塞等待vs非阻塞等待

阻塞等待:子进程不退出,父进程就一直等待,直到子进程退出后wait/waitpid函数才能返回,期间父进程不能做任何事。

非阻塞等待:子进程不退出,waitpid直接返回,通常需要重复调用,使用非阻塞轮询方案来进行等待,在重复调用的期间,父进程可以做自己占据时间不多的事情。

下面使用非阻塞等待来测试一下非阻塞等待的性质,下面的代码中我只进行一次非阻塞等待,通过下图我们发现waitpid只执行了一次就返回了,并且之后父进程也是直接退出了,通过观察子进程的ppid我们发现子进程变成了孤儿进程,所以非阻塞等待通常需要重复进行。

#include     
#include     
#include     
#include     
#include     
    
void Worker()    
{    
    int cnt = 3;    
    while(cnt)    
    {    
        printf("I am child process , pid: %d , ppid : %d , cnt : %d
" ,getpid(),getppid(),cnt--);        
        sleep(1);    
    }    
}    
    
    
int main()    
{    
    pid_t id = fork();                                                    
    
    if(id == 0)    
    {
        // child    
        Worker();    

        exit(0);
    }
    else 
    {
        // 获取子进程退出信息
        int status = 0;
        pid_t rid = waitpid(id,&status,WNOHANG);

        if(rid > 0) // 等待成功,且子进程退出
        {
            printf("child quit success , exit code : %d , exit sig : %d
",status>>8,status&0x7F);
        }
        else if(rid == 0) // 等待成功,但子进程未退出,重复等待
        {
            printf("child is alive , wait again , father do other thing...
");
        }
        else // rid < 0 ,等待失败,通常是id出错导致的 
        {
            printf("wait failed!
");
        }
        
        sleep(1);
    }

    return 0;
}

在讲解非阻塞等待时,父进程可以做自己占据时间不多的事情,下面设计一个非阻塞轮询等待的代码来实现这个功能。

#include 
#include 
#include 
#include 
#include 

#define TASK_NUM 5
typedef void(*task_t)();

//
void download()
{
    printf("this is a download task is running!
");    
}    
    
void printLog()    
{    
    printf("this is a write log task is running!
");    
}    
    
void show()    
{    
    printf("this is a show task is running!
");    
}    
//
// 初始化任务表    
void InitTasks(task_t tasks[],int num)    
{    
    int i = 0;     
    for(; i < num ; i++)                                                                                                                                 
    {    
        tasks[i] = NULL;    
    }    
}    

// 向任务表中添加任务
int AddTask(task_t tasks[], int num ,task_t task)
{
    int i = 0; 
    for( ; i < num ; i++)
    {
        if(tasks[i] == NULL)                                                                                                                             
        {
            tasks[i] = task;
            return 1;
        }
    }
    return 0;
}

// 该函数内部使用了回调式执行任务
// 可以在函数的内部自行增加或移除任务
void executeTask(task_t tasks[] , int num)
{
    int i = 0;
    for(; i < num ; i++)
    {
        if(tasks[i])    tasks[i]();
    }
}

// 为子进程调用设计的函数
void Worker()
{
    int cnt = 3;
    while(cnt)
    {
        printf("I am child process , pid: %d , ppid : %d , cnt : %d
" ,getpid(),getppid(),cnt--);
        sleep(1);
    }
}
int main()
{
    // 定义一个任务表
    task_t tasks[TASK_NUM];
                                                                                                                                                         
    // 初始化并添加任务到任务表中
    InitTasks(tasks,TASK_NUM);
    AddTask(tasks,TASK_NUM,download);
    AddTask(tasks,TASK_NUM,printLog);
    AddTask(tasks,TASK_NUM,show);

    pid_t id = fork();
    
    if(id == 0)
    {
        // child
        Worker();

        exit(0);
    }
    else 
    {
        // 非阻塞轮询等待
        while(1)
        {
            // 获取子进程退出信息
            int status = 0;
            pid_t rid = waitpid(id,&status,WNOHANG);

            if(rid > 0) // 等待成功,且子进程退出
            {
                printf("child quit success , exit code : %d , exit sig : %d
",status>>8,status&0x7F);
                break;
            }
            else if(rid == 0) // 等待成功,但子进程未退出,重复等待
            {
                printf("------------------------------------------------------
");
                printf("child is alive , wait again , father do other thing...
");
                executeTask(tasks,TASK_NUM);
                printf("------------------------------------------------------
");
            }
            else // rid < 0 ,等待失败,通常是id出错导致的 
            {
                printf("wait failed!
");
                break;
            }
            
            sleep(1);
        }
    }

    return 0;
}


3.3.2.4 返回值

waitpid函数的返回值:

  • <0,等待失败
  • ==0,等待成功,但子进程还未退出
  • >0,等待成功,且子进程退出

3.3.3 操作系统层面上父进程是如何获取子进程的退出信息

上面讲到了父进程使用系统调用来获取子进程的退出信息,那么接下来讲解一下在操作系统层面上父进程是如何获取子进程的退出信息的。

父进程和子进程分别有自己的PCB,父进程等待子进程的本质就是调用wait/waitpid函数来等待子进程,在父进程中定义一个变量 int status,在调用这两个函数时,需要将status的地址作为参数传给这两个函数,这里设参数名为statusp,statusp就指向了status。当子进程退出时,子进程的代码和数据就会被销毁,代码中main函数中有的return和exit,操作系统执行退出逻辑,会将子进程的退出信息写入到子进程的PCB中,所以父进程在调用wait/waitpid函数时,这两个函数底层就是将子进程的状态修改为僵尸状态,并将信号编号和退出码组合起来存储在*statusp中,*statusp = (exit_code<<8)|exit_sig,这里的*statusp也就是父进程中的status,所以父进程可以通过wait/waitpid这两个函数来获取子进程的信号编号和退出信息。


3.4 父进程是在子进程的等待队列中等待的

我们在讲阻塞时讲到过进程可能会访问操作系统中的底层硬件设备,操作系统为了管理这些设备都要为其创建对应的结构体对象,当时讲到一个进程在设备上进行等待,本质上是这个设备为进程提供了等待队列,那么这里父进程等待子进程本质上同样也是子进程为父进程提供了等待队列,当子进程结束后,操作系统就会从子进程的等待队列中将父进程拿出来,再放入运行队列中,调度父进程就完成了父进程的等待过程,所以我们要记住只有进程是阻塞状态,那么这个进程一定需要被放入到某个等待队列中。


3.5 父进程等待多个子进程

下面设计了一个代码,父进程按0 ~ 5顺序创建6个子进程,设计一个Worker函数参数为子进程的顺序编号,子进程调用Worker函数会将对应子进程的编号输出,将子进程的顺序编号传入到exit函数中,然后父进程调用6次waitpid函数进行等待,等待成功会输出子进程的退出码,运行代码并启动进程监控脚本我们可以发现,父进程一下子创建了6个进程,根据代码输出的结果来看也确实是创建了6个子进程,这些子进程并不是按照0 ~ 5这个顺序进行调用的,等待时也不是按照这个顺序等待的。

#include     
#include     
#include     
#include     
#include     
    
void Worker()    
{    
    int cnt = 3;    
    while(cnt)    
    {    
        printf("I am child process , pid: %d , ppid : %d , cnt : %d , number : %d
" ,getpid(),getppid(),cnt--,number);
        sleep(1);    
    }    
}    
    
#define n 6    
    
int main()    
{    
    for(int i = 0 ; i < n ; i++)    
    {    
        pid_t id = fork();    
    
        if(id == 0)    
        {    
            // child    
            Worker(number);    
    
            exit(i);                                                                                                                                     
        }    
    
    }    
    
    for(int i = 0 ; i < n ; i++)
    {
    	int status = 0;
		pid_t rid = waitpid(-1,&status,0);
		
		if(rid > 0)
		{
	        printf("wait child %d success , exit code : %d
",rid,WEXITSTATUS(status));                                                              
	    }
	}

    return 0;
}


四、进程替换

到目前为止我们创建子进程,执行的代码都是父进程代码的一部分,如果我们想让子进程执行新的程序应该怎么办呢?

那就要讲到进程替换了,进程替换能够让正在运行的进程变为运行另一个我们指定进程,所以也能子进程执行全新的代码和访问全新的数据,让子进程与父进程再无瓜葛。

4.1 替换函数

4.1.1 七种替换函数

库函数中有六种exec开头的函数,统称exec函数,这些函数实现的都是同一个功能,设计这么多函数的目的就是为了满足各种调用场景。

#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 execvpe(const char *file, char *const argv[],char *const envp[]);

系统调用中有一个进程替换的函数,上面的六个函数底层都是封装了execve函数的。

int execve(const char *path, char *const argv[], char *const envp[]);

4.1.2 函数解释

  • 这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回。
  • 如果调用出错则返回-1
  • 所以exec函数只有出错的返回值而没有成功的返回值。

4.1.3 命名理解

这些函数原型看起来很容易混,但只要掌握了规律就很好记。

  • l(list) : 表示参数采用列表
  • v(vector) : 参数用数组
  • p(path) : 有p自动搜索环境变量PATH
  • e(env) : 表示自己维护环境变量

4.2 进程替换的使用

4.2.1 单进程版程序替换的使用

我们编写一个代码,让这个代码运行起来后,进行进程替换,执行操作系统中的ls指令,在进程替换的前后分别添加两句输出语句,运行这个代码观察现象,通过下图我们发现确实运行了ls指令,但是原来的进程中只打印了进程替换之前的输出语句,而替换后的语句没有打印出来。

#include     
#include     
    
int main()    
{    
    printf("pid : %d , exec command begin
",getpid());    
    
    execl("/usr/bin/ls","ls","-a","-l",NULL);                                                                                                            
    
    printf("pid : %d , exec command end
",getpid());    
    
    return 0;    
} 


4.2.2 多进程版程序替换的使用

这里我们编写一个代码,让父进程创建一个子进程,子进程在运行的过程中,将子进程替换为操作系统中的ls指令,在进程替换的前后分别添加输出语句,让父进程以阻塞等待的方式等待子进程,运行这个代码观察现象,通过下图我们发现确实运行了ls指令,但是原来的进程中只打印了进程替换之前的输出语句,而替换后的语句没有打印出来,并且我们发现子进程的输出的pid与父进程中waitpid的返回值相同,所以子进程在发生进程替换的时候并不会改变子进程的pid。

int main()                                                                                                                                               
{     
    pid_t id = fork();  
  
    if(id == 0)  
    {  
        //child  
           
        printf("pid : %d , exec command begin
",getpid());  
  
        execl("/usr/bin/ls","ls","-a","-l",NULL);  
  
        printf("pid : %d , exec command end
",getpid());  
    }  
    else   
    {  
        // father  
        pid_t rid = waitpid(-1,NULL,0);  
        if(rid > 0)  
        {  
            printf("Wait Success , rid : %d
",rid);  
        }                                                                           
    }                                                                               
                                                                                    
    return 0;    
} 


4.2.3 各种替换函数在多进程的使用

在执行exec*函数时,必须要解决下面两个问题:

  1. 必须先找到这个可执行程序
  2. 必须告诉exec*函数需要怎么执行
4.2.3.1 execl函数
int execl(const char *path, const char *arg, ...);

函数名中字符的含义:
函数名中的l(list)代表这个函数的传参方式为列表方式

参数:

  • path:目标可执行程序的路径和文件名

  • arg:传递给新程序的参数列表,arg 必须是参数列表的第一个元素,通常设为新程序的名称

  • :代表可变类型参数列表,可以传任意数量的额外参数给新进程,通常传这个新进程的执行选项,这些参数将作为新进程的命令行参数,可变类型参数列表必须以NULL结尾代表传参结束

int main()                                                                                                                                               
{     
    pid_t id = fork();  
  
    if(id == 0)  
    {  
        //child  
           
        printf("pid : %d , exec command begin
",getpid());  
  
        execl("/usr/bin/ls","ls","-a","-l",NULL);  
  
        printf("pid : %d , exec command end
",getpid());  
    }  
    else   
    {  
        // father  
        pid_t rid = waitpid(-1,NULL,0);  
        if(rid > 0)  
        {  
            printf("Wait Success , rid : %d
",rid);  
        }                                                                           
    }                                                                               
                                                                                    
    return 0;    
} 


4.2.3.2 execlp函数
int execlp(const char *file, const char *arg, ...);

函数名中字符的含义:

  • 函数名中的l(list)代表这个函数的传参方式为列表方式
  • 函数名中的p(path)代表函数会自动搜索环境变量PATH查找file的路径

参数:

  • file:目标可执行程序的文件名

  • arg:传递给新程序的参数列表,arg 必须是参数列表的第一个元素,通常设为新程序的名称

  • :代表可变类型参数列表,可以传任意数量的额外参数给新进程,通常传这个新进程的执行选项,这些参数将作为新进程的命令行参数,可变类型参数列表必须以NULL结尾代表传参结束

int main()    
{       
    pid_t id = fork();       
      
    if(id == 0)                                          
    {                                                     
        //child                                     
                                            
        printf("pid : %d , exec command begin
",getpid());      
      
        execlp("ls","ls","-a","-l",NULL);                                                                                                                                  
                                                             
        printf("pid : %d , exec command end
",getpid());      
    }                               
    else                            
    {                               
        // father                          
        pid_t rid = waitpid(-1,NULL,0);      
        if(rid > 0)      
        {      
            printf("Wait Success , rid : %d
",rid);      
        }      
    }      
      
    return 0;      
}      


4.2.3.3 execv函数
int execv(const char *path, char *const argv[]);

函数名中字符的含义:

  • 函数名中的v(vector)代表这个函数的传参方式为数组方式

参数:

  • path:目标可执行程序的路径和文件名
  • argv:代表进程需要将新进程的执行方式以字符串的方式存入一个字符串数组中,并将这个字符串数组传给argv
int main()
{                         
  	char* const argv[] = {"ls" , "-a" , "-l" , NULL};
                   
    pid_t id = fork();
                   
    if(id == 0)
    {                                                          
        //child
         
        printf("pid : %d , exec command begin
",getpid());
                                                       
        execv("/usr/bin/ls",argv);                                                                                                                     
            
        printf("pid : %d , exec command end
",getpid());
    }            
    else                               
    {              
        // father
        pid_t rid = waitpid(-1,NULL,0);             
        if(rid > 0)
        {
            printf("Wait Success , rid : %d
",rid);
        }    
    }

    return 0;
}


4.2.3.4 execvp函数
int execvp(const char *file, char *const argv[]);

函数名中字符的含义:

  • 函数名中的v(vector)代表这个函数的传参方式为数组方式
  • 函数名中的p(path)代表函数会自动搜索环境变量PATH查找file的路径

参数:

  • file:目标可执行程序的文件名
  • argv:代表进程需要将新进程的执行方式以字符串的方式存入一个字符串数组中,并将这个字符串数组传给argv
int main()
{                         
  	char* const argv[] = {"ls" , "-a" , "-l" , NULL};
                   
    pid_t id = fork();
                   
    if(id == 0)
    {                                                          
        //child
         
        printf("pid : %d , exec command begin
",getpid());
                                                       
        execvp("ls",argv);  
        // 也可以使用下面这种写法,数组下标0位置处也是文件名
        // execvp("argv[0]",argv);                                                                                                                      
            
        printf("pid : %d , exec command end
",getpid());
    }            
    else                               
    {              
        // father
        pid_t rid = waitpid(-1,NULL,0);             
        if(rid > 0)
        {
            printf("Wait Success , rid : %d
",rid);
        }    
    }

    return 0;
}


4.2.3.5 execle函数
int execle(const char *path, const char *arg, ...,char *const envp[]);

函数名中字符的含义:

  • 函数名中的l(list)代表这个函数的传参方式为列表方式
  • 函数名中的e(env) 代表该进程自己维护环境变量

参数:

  • path:目标可执行程序的路径和文件名

  • arg:传递给新程序的参数列表,arg 必须是参数列表的第一个元素,通常设为新程序的名称

  • :代表可变类型参数列表,可以传任意数量的额外参数给新进程,通常传这个新进程的执行选项,这些参数将作为新进程的命令行参数,可变类型参数列表必须以NULL结尾代表传参结束

// test.cc
#include     
    
using namespace std;    
    
int main(int argc , char* argv[] , char* env[])    
{    
    for(int i = 0 ; i < argc ; i++)    
    {    
        cout << i << " : " << argv[i] << endl;    
    }    
    
    for(int i = 0 ; env[i] ; i++)    
    {    
        cout << i << " : " << env[i] << endl;                                                                                                            
    }    
    
    
    return 0;    
} 
// process.c
extern char ** environ;    
    
int main()                                               
{                                                        
	char* const argv[] = {"ls" , "-a" , "-l" , NULL};                                                                                                  
                                                                                                     
    pid_t id = fork();                                                                               
                                                                                                     
    if(id == 0)                                                                                      
    {                                                                                                
        //child                                                                                      
                                                                                                     
        printf("pid : %d , exec command begin
",getpid());                                          
        execle("./mytest","mytest","-a","-b",NULL,environ);                                                                                       
                                                                                                     
        printf("pid : %d , exec command end
",getpid());                                            
    }                                                                                                
    else                                                                                             
    {    
        // father    
        pid_t rid = waitpid(-1,NULL,0);    
        if(rid > 0)    
        {    
            printf("Wait Success , rid : %d
",rid);    
        }    
    }    
    
    return 0;    
}    


4.2.3.6 execvpe函数
int execvpe(const char *file, char *const argv[],char *const envp[]);

函数名中字符的含义:

  • 函数名中的v(vector)代表这个函数的传参方式为数组方式
  • 函数名中的p(path)代表函数会自动搜索环境变量PATH查找file的路径
  • 函数名中的e(env) 代表该进程自己维护环境变量

参数:

  • file:目标可执行程序的文件名
  • argv:代表进程需要将新进程的执行方式以字符串的方式存入一个字符串数组中,并将这个字符串数组传给argv
  • envp:代表进程需要将新进程所需要的环境变量存储到一个字符串数组中,并将这个字符串数组传给envp
extern char ** environ;    
    
int main()    
{    
 	char* const argv[] = {"ls" , "-a" , "-l" , NULL};    
    
    pid_t id = fork();    
    
    if(id == 0)    
    {    
        //child    
    
        printf("pid : %d , exec command begin
",getpid());    
        execvpe("ls",argv,environ);                                                                                                                    

        printf("pid : %d , exec command end
",getpid());    
    }    
    else     
    {    
        // father    
        pid_t rid = waitpid(-1,NULL,0);    
        if(rid > 0)    
        {    
            printf("Wait Success , rid : %d
",rid);    
        }    
    }    
    
    return 0;    
}    


4.2.4 进程替换可以替换各种语言的进程

上面对进程替换函数进行了使用,发现进程替换可以替换操作系统的指令,实际上不仅仅是可以替换操作系统中的指令,只要是能够运行起来变为进程的任何语言的程序都可以被替换,例如我们写的C/C++程序,Python程序,脚本程序都可以被替换,因为系统大于一切。


4.2.5 进程替换中子进程获取环境变量

  1. 当我们进行程序替换的时候,子进程对应的环境变量,是可以直接从父进程来的,我们在环境变量那篇文章中讲到过,这里不做过多讲解。

  2. 环境变量被子进程继承下去是一种默认行为,不受程序替换的影响,创建子进程,子进程的进程地址空间都是复制父进程的,发生进程替换时,新进程的代码和数据会替换原进程的代码和数据,但是不会替换环境变量。

  3. 让子进程执行的时候获取环境变量

    • 将父进程的环境变量原封不动的传给子进程

      • 子进程可以天然的获取父进程的环境变量
      • 再使用进程替换函数时,将父进程的环境变量作为参数传给子进程
    • 我们想传递我们自己的环境变量,可以直接构造一个环境变量表,给子进程传递,需要注意的是子进程可以直接获取父进程的环境变量,若我们将自己的环境变量表作为参数传给进程替换函数时,不是在父进程环境变量的基础上新增环境变量表,而是直接将我们的环境变量表对子进程获取到父进程的环境变量进行覆盖,也就是说子进程不会拥有父进程的环境变量,只会拥有我们自己的环境变量

    • 使子进程的环境变量在原父进程环境变量的基础上新增环境变量,我们可以在父进程中使用putenv函数在父进程中添加环境变量,子进程自然就可以获取到这些新添加的环境变量


4.3 进程替换的原理

4.3.1 单进程替换的原理

进程有自己独立的PCB、进程地址空间和页表,虚拟内存通过页表映射到物理内存中,进程在物理内存中有自己的代码和数据,当进程进行进程替换时,操作系统会将新进程的代码和数据从磁盘中取出,并将原进程的代码和数据进行覆盖,当进程替换完后,进程被调度,那么进程执行的就是新进程的代码和数据。


4.3.2 多进程替换的原理

父进程创建子进程以后,父子进程分别有自己的独立的PCB、进程地址空间和页表,但是父子进程的代码和数据是共享的,所以父/子进程进行进程替换时,会发生写时拷贝,在物理内存中将代码和数据再重新创建一份,将新进程的代码和数据替换掉调用exec*函数的进程中的代码和数据。


4.3.3 小知识点

  1. exec* 这样的函数只要调用成功,那么原进程的后序代码就没有机会再执行了,因为原进程的代码和数据会被新进程替换。

  2. exec* 这样的函数只有失败的时候有返回值,成功时没有返回值,但是通常使用的时候都不会判断返回值,因为函数出错了就会执行原进程的代码。

  3. 在进程替换的过程中,只是将代码和数据进行替换,所以进程的pid不会改变。在多进程关系中,发生进程替换,父子进程的父子关系也不会改变

  4. 这里大家或许有疑问,被替换后的进程怎么知道要从最开始执行,它是如何知道最开始的地方在哪里的呢?

    在Linux操作系统中,可执行程序是有格式的(ELF),可执行程序中的头部有一个字段entry,entry记录的是可执行程序的入口地址。


4.3.4 进程替换与程序加载到内存的关系

我们之前学习过程序加载到内存是什么?为什么?这里我们需要怎么将程序加载到内存呢?

程序加载到内存是什么?
指将程序从硬盘读取并放置到计算机的内存中以便执行的过程。

程序加载到内存为什么?
是因为冯诺依曼体系规定的,内存的访问速度远远快于硬盘等存储介质,将程序加载的内存能够提高了程序的执行速度。

如何将程序加载到内存?
将程序加载到内存实际上就是将进程的代码和数据加载到内存中,而我们学习到的进程替换就能够将进程的代码和数据加载到内存中,实际上加载进程中就使用了进程替换,进程替换就是加载器中非常重要的一部分。


结尾

如果有什么建议和疑问,或是有什么错误,大家可以在评论区中提出。
希望大家以后也能和我一起进步!!🌹🌹
如果这篇文章对你有用的话,希望大家给一个三连支持一下!!🌹🌹

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

搜索文章

Tags

docker 容器 运维 java-rabbitmq java PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 开源 自动化 n8n dity make 服务器 linux rag ragflow ollama 大模型部署 人工智能 ubuntu linux环境变量 搜索引擎 程序员 大模型应用 prompt LLM 大模型 pytorch tensorflow 大模型面经 职场和发展 Deepseek 大模型学习 数据库 postgresql pgpool centos 数据分析 机器学习 计算机视觉 嵌入式硬件 边缘计算 qt linuxdeployqt 打包部署程序 appimagetool AI Dify Agent DeepSeek llama CrewAI python chatgpt gpu算力 安全 uniapp 网络 vue android studio 交互 后端 websocket android tcp/ip ai nlp 无人机 机器人 Linux DNS flutter Google pay Apple pay 硬件工程 单片机 物联网 区块链 分布式账本 web3 智能合约 信任链 共识算法 milvus 知识图谱 大模型教程 AI大模型 deepseek 科技 大数据 云计算 debian wps vscode 安卓 Linux 维护模式 相机 pycharm flask macos MacMini Mac 迷你主机 mini Apple 智能手机 计算机网络 web安全 网络安全 php 技能大赛 YOLO 深度学习 网络协议 ide github ffmpeg 开发语言 ESXi opencv onlyoffice 在线office 操作系统 nac 802.1 portal Qwen3 qwen3 32b vllm 本地部署 gitlab 远程连接 MQTT mosquitto 消息队列 vim c++ 系统架构 nginx chrome 学习 阿里云 spring boot spring 微信开放平台 微信公众平台 微信公众号配置 华为 智能路由器 笔记 WSL2 上安装 Ubuntu vmware 虚拟机 windows oracle c语言 区块链项目 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 redis 分布式 缓存 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 网页服务器 web服务器 Nginx ssh vue.js react.js 前端 云原生 iventoy VmWare OpenEuler HTTP 服务器控制 ESP32 DeepSeek rocketmq 远程 命令 执行 sshpass 操作 政务 分布式系统 监控运维 Prometheus Grafana top Linux top top命令详解 top命令重点 top常用参数 fiddler 爬虫 http conda mysql 主从复制 Apache Beam 批流统一 性能优化 案例展示 数据分区 容错机制 kubernetes go GIS 遥感 WebGIS 多线程服务器 TCP服务器 qt项目 qt项目实战 qt教程 BMC IPMI 带外管理 tidb GLIBC mamba 服务器扩容没有扩容成功 harmonyos 编辑器 鸿蒙 鸿蒙系统 前端框架 车载系统 金融 算法 数据挖掘 kylin ruoyi oceanbase rc.local 开机自启 systemd 麒麟 mcu ssl fpga开发 virtualenv arm 线程 多线程 ShapeFile GeoJSON Python 卸载 软件 列表 环境迁移 Windows ai工具 存储维护 NetApp存储 EMC存储 mcp mcp协议 go-zero mcp服务器 串口服务器 万物互联 工业自动化 工厂改造 k8s django ROS 自动驾驶 c# 大模型推理 MCP AIGC ansible playbook 自动化运维 架构 语言模型 华为云 华为od udp 经验分享 kvm qemu libvirt eureka spring cloud 蓝桥杯 Dell HPE 联想 浪潮 AI-native elasticsearch 7-zip 关系型 UEFI Legacy MBR GPT U盘安装操作系统 jvm 游戏 程序人生 wsl2 wsl vsxsrv Python教程 pycharm安装 目标检测 YOLOv12 git maven 音视频 学习方法 神经网络 卷积神经网络 servlet muduo 网络库 hadoop big data 考研 数据结构 ddos 具身智能 强化学习 网络结构图 yaml Ultralytics 可视化 ssh漏洞 ssh9.9p2 CVE-2025-23419 ragflow 源码启动 游戏引擎 jenkins 1024程序员节 agi pip Kali 渗透 部署 javascript edge 网工 运维开发 Deepseek-R1 私有化部署 推理模型 p2p CH340 串口驱动 CH341 uart 485 Cursor appium 软件测试 自动化测试 功能测试 程序 编程 内存 性能分析 IO 自然语言处理 H3C 驱动开发 cursor kafka https tomcat 其他 智慧农业 开源鸿蒙 团队开发 嵌入式 linux驱动开发 arm开发 webpack 课程设计 pyautogui rpa llm 群晖 低代码 ACL 流量控制 基本ACL 网络管理 规则配置 云电竞 云电脑 todesk ipython visualstudio r语言 数据可视化 word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 微服务 源码 毕业设计 RAGFlow Ollama excel CosyVoice 企业微信 vmamba 小程序 hdc 鸿蒙NEXT 图文教程 VMware虚拟机 macOS系统安装教程 macOS最新版 虚拟机安装macOS Sequoia Vmamba eclipse prometheus grafana intellij-idea 网络工程师 软考 2024 2024年上半年 下午真题 答案 AI编程 visual studio code 镜像源 transformer 大模型入门 AI员工 Playwright pythonai PlaywrightMCP gpt mysql离线安装 ubuntu22.04 mysql8.0 harmonyOS面试题 中兴光猫 换光猫 网络桥接 自己换光猫 SenseVoice bash 负载均衡 fstab rabbitmq ruby 计算机外设 bug mysql安装报错 windows拒绝安装 gcc g++ g++13 grep 智能驾驶 BEVFusion Ubuntu 面试 unity springsecurity6 oauth2 授权服务器 前后端分离 yolov5 Claude Desktop Claude MCP Windows Cli MCP devops LSTM 信息与通信 FTP服务器 自定义客户端 SAS LLM Web APP Streamlit vue3 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 gaussdb jupyter 鲲鹏 昇腾 npu uni-app 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 PyQt PySide6 sdkman typescript 重启 排查 系统重启 日志 原因 notepad burpsuite 安全工具 mac安全工具 burp安装教程 渗透工具 adb Ubuntu20.04 2.35 cnn GoogLeNet pdf 办公自动化 自动化生成 pdf教程 5G 3GPP 卫星通信 电脑 镜像 docker-compose 图形化界面 vite powerpoint Docker Docker Compose Kubernetes 压力测试 大模型压力测试 EvalScope Apache Flume 数据采集 安装部署 配置优化 高级功能 大数据工具集成 网络攻击模型 系统安全 v10 DeepSeek-R1 API接口 armbian u-boot oneapi svn jmeter ESP32 matlab C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 webrtc IM即时通讯 QQ 微信 剪切板对通 HTML FORMAT word yolov8 gnu zabbix 远程工作 mac 截图 录屏 gif 工具 docker run 数据卷挂载 环境变量 端口映射 交互模式 知识库 本地知识库部署 DeepSeek R1 模型 Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 QT 5.12.12 QT开发环境 Ubuntu18.04 gitee node.js 集成学习 集成测试 抓包工具 框架搭建 前端面试题 持续部署 selenium 测试工具 网络爬虫 cron crontab日志 工具分享 腾讯云 tcpdump LVM 磁盘分区 lvresize 磁盘扩容 pvcreate android-studio SSE stm32 远程桌面 服务器配置 数据库系统 我的世界服务器搭建 游戏服务器 Minecraft 飞腾处理器 硬件架构 国产化 gitee go 微信小程序 notepad++ 模拟器 教程 键盘 卡死 list 安卓模拟器 隐藏文件 kotlin iphone openssl unix ArkUI ArkTS 移动端开发 iTerm2 VMware 软件需求 设备树 毕设 Linux find grep MobaXterm 文件传输 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 iftop 网络流量监控 视频编解码 vnc ui axure 富文本编辑器 Alexnet nvm 网络药理学 生信 分子对接 autodock mgltools PDB PubChem VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 防火墙 端口号 开放端口 访问列表 智能体开发 lvs sql golang audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 YOLOv8 NPU Atlas800 A300I pro asi_bench 模型联网 API CherryStudio sqlserver iot gitea IIS服务器 IIS性能 日志监控 模拟退火算法 数据库架构 数据管理 数据治理 数据编织 数据虚拟化 安全漏洞 信息安全 shell NVML nvidia-smi chatbox chrome devtools chromedriver GCC crosstool-ng shell脚本免交互 expect linux免交互 maxkb ARG open webui arkUI arkTs nohup 异步执行 python3.11 openjdk comfyui comfyui教程 CUPS 打印机 Qt5 WSL2 IP 地址 AnythingLLM AnythingLLM安装 arcgis 人工智能生成内容 C++ 7z Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 计算生物学 生物信息学 生物信息 基因组 opengl ubuntu20.04 ros ros1 Noetic 20.04 apt 安装 DevOps 软件交付 数据驱动 应用场景 数据安全 qps 高并发 jdk RTSP xop RTP RTSPServer 推流 视频 Trae IDE AI 原生集成开发环境 Trae AI fastapi 回显服务器 UDP的API使用 华为认证 交换机 dell服务器 Flask FastAPI Waitress Gunicorn uWSGI Uvicorn 银河麒麟 npm rsyslog kind json html5 firefox triton 模型分析 micropython esp32 mqtt GPU 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 vu大文件秒传跨域报错cors 语音识别 蓝桥杯C++组 deepseek-r1 大模型本地部署 ftp服务 文件上传 apache web 框架 microsoft 匿名FTP 邮件传输代理 SSL支持 chroot监狱技术 Java进程管理 DevOps自动化 脚本执行 跨平台开发 远程运维 Apache Exec JSch pygame 小游戏 五子棋 ip chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 kali 共享文件夹 虚拟显示器 远程控制 Mac内存不够用怎么办 开机黑屏 xcode 程序员创富 diskgenius rime 进程间通信 图像处理 策略模式 zip unzip IMX317 MIPI H265 VCU Python基础 Python技巧 实时音视频 实时互动 camera Arduino 电子信息 gromacs 分子动力学模拟 MD 动力学模拟 ip协议 单例模式 html kernel k8s部署 MySQL8.0 高可用集群(1主2从) 显卡驱动持久化 GPU持久化 DrissionPage SRS 流媒体 直播 硬件 设备 PCI-Express 升级 CVE-2024-7347 漏洞 个人博客 大模型微调 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 雨云 NPS IIS .net core Hosting Bundle .NET Framework vs2022 AutoDL live555 rtsp rtp VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 xrdp 安全威胁分析 宝塔面板 同步 备份 建站 反向代理 RAID RAID技术 磁盘 存储 监控 我的世界 我的世界联机 数码 源码剖析 rtsp实现步骤 流媒体开发 rust腐蚀 file server http server web server EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 aws googlecloud 银河麒麟操作系统 大语言模型 langchain NFS redhat 状态管理的 UDP 服务器 Arduino RTOS .net db jar Windsurf 宝塔 oracle fusion oracle中间件 NFC 近场通讯 智能门锁 Chatbox Qualcomm WoS QNN AppBuilder 虚拟化 EtherCAT转Modbus EtherCAT转485网关 ECT转485串口服务器 ECT转Modbus485协议 ECT转Modbus串口网关 ECT转Modbus串口服务器 es6 qt6.3 g726 华为OD 华为OD机试真题 可以组成网络的服务器 css css3 电路仿真 multisim 硬件工程师 硬件工程师学习 电路图 电路分析 仪器仪表 Python 视频爬取教程 Python 视频爬取 Python 视频教程 localhost 进程 进程控制 C 进程地址空间 邮件APP 免费软件 嵌入式Linux IPC xshell termius iterm2 客户端 MySql java-ee MacOS 向日葵 python2 ubuntu24.04 桌面环境 Isaac Sim 虚拟仿真 export env 变量 chromium dpi 打不开xxx软件 无法检查其是否包含恶意软件 IP配置 netplan SecureCRT lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 cuda Bug解决 Qt platform OpenCV Maxkb RAG技术 本地知识库 OpenCore 权限 鸿蒙面试 面试题 热榜 rpc apt 国内源 rnn 多进程 双系统 GRUB引导 Linux技巧 yum换源 WebVM centos 7 flash-attention 报错 国产数据库 瀚高数据库 数据迁移 下载安装 dify 本地化部署 计算机 cudnn nvidia dash 正则表达式 remote-ssh 进程信号 命名管道 客户端与服务端通信 post.io 企业邮箱 搭建邮箱 单元测试 可用性测试 小智 京东云 deepseek-v3 ktransformers RagFlow RAG Cline spark HistoryServer Spark YARN jobhistory FTP 服务器 ftp RustDesk自建服务器 rustdesk服务器 docker rustdesk ping++ 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 react next.js 部署next.js 孤岛惊魂4 媒体 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 adobe 国产操作系统 open Euler dde deepin 统信UOS windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 express 测试用例 游戏程序 三级等保 服务器审计日志备份 DOIT 四博智联 服务器繁忙 备选 网站 api 调用 示例 大数据平台 CDN rustdesk 镜像下载 freebsd lighttpd安装 Ubuntu配置 Windows安装 性能测试 服务器优化 postman 虚幻 信号处理 跨域请求 .netcore 宠物 免费学习 宠物领养 宠物平台 mongodb yum RAGFLOW 检索增强生成 文档解析 大模型垂直应用 make命令 makefile文件 ios USB转串口 PX4 cocoapods Qwen2.5-coder 离线部署 版本 安装 nohup后台启动 高德地图 鸿蒙接入高德地图 HarmonyOS5.0 WSL resolv.conf 快捷键 旋转屏幕 自动操作 Ubuntu 22.04 算家云 算力租赁 安装教程 termux 环境搭建 mac设置host 虚拟现实 向量数据库 milvus安装 csrf vm Xinference 数据集 影刀 #影刀RPA# 腾讯云大模型知识引擎 Docker Desktop 产测工具框架 IMX6ULL 管理框架 elk Logstash 日志采集 拓扑图 网络用户购物行为分析可视化平台 大数据毕业设计 软链接 硬链接 wireshark 自动化任务管理 虚拟局域网 seleium 3d 飞牛nas fnos ip命令 新增网卡 新增IP 启动网卡 docker desktop image CPU 使用率 系统监控工具 linux 命令 tar 安装MySQL rtc xml 计算机学习路线 编程语言选择 easyTier 内网穿透 组网 论文笔记 C语言 Kylin-Server 服务器安装 Invalid Host allowedHosts 个人开发 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 opcua opcda KEPServer安装 做raid 装系统 ssrf 失效的访问控制 IMM web3.py Headless Linux string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap kamailio sip VoIP WebRTC opensearch helm 项目部署到linux服务器 项目部署过程 火绒安全 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 创意 社区 zookeeper asp.net大文件上传 asp.net大文件上传下载 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 云服务 midjourney AI写作 asm ABAP 毕昇JDK llama.cpp 机柜 1U 2U Pyppeteer numpy 大模型训练/推理 推理问题 mindie c/s 密码学 哈希算法 minio 文件存储服务器组件 #STC8 #STM32 云计算面试题 GPU状态 私有化 bcompare Beyond Compare GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 软件工程 稳定性 看门狗 设计模式 软件卸载 系统清理 并查集 leetcode 通信工程 毕业 英语 本地环回 bind jellyfin nas EVE-NG 终端工具 远程工具 ollama api ollama外网访问 GPU训练 裸机装机 linux磁盘分区 裸机安装linux 裸机安装ubuntu 裸机安装kali 裸机 华为鸿蒙系统 ArkTS语言 Component 生命周期 条件渲染 Image图片组件 Obsidian Dataview MAC Kali Linux 蓝耘科技 元生代平台工作流 ComfyUI 工作流 agent workflow 实习 Claude 王者荣耀 okhttp efficientVIT YOLOv8替换主干网络 TOLOv8 searxng AI Agent 字节智能运维 webdav 目标跟踪 OpenVINO 推理应用 性能调优 安全代理 nftables scapy Linux Vim ci/cd IPv4/IPv6双栈 双栈技术 网路规划设计 ensp综合实验 IPv4过渡IPv6 IPv4与IPv6 磁盘挂载 新盘添加 partedUtil 隐藏目录 文件系统 管理器 通配符 rust 软考设计师 中级设计师 SQL 软件设计师 linux子系统 忘记密码 gru MAVROS 四旋翼无人机 Apache OpenNLP 句子检测 分词 词性标注 核心指代解析 mcp-proxy mcp-inspector fastapi-mcp sse Linux网络编程 MCP server C/S EMQX 通信协议 pillow vscode1.86 1.86版本 ssh远程连接 阻塞队列 生产者消费者模型 服务器崩坏原因 常用命令 文本命令 目录命令 NAT转发 NAT Server UDP EasyConnect LDAP cpu 实时 使用 单一职责原则 统信 虚拟机安装 echarts 信息可视化 网页设计 NAS Termux Samba Hyper-V WinRM TrustedHosts 需求分析 规格说明书 frp 内网服务器 内网代理 内网通信 LLMs 黑客 渗透测试 信息收集 直播推流 网络编程 聊天服务器 套接字 TCP Socket 大文件秒传跨域报错cors 大大通 第三代半导体 碳化硅 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 FunASR ASR 算力 rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 阿里云ECS ECT转Modbus协议 EtherCAT转485协议 ECT转Modbus网关 C# MQTTS 双向认证 emqx 浏览器自动化 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 hive outlook 错误代码2603 无网络连接 2603 gunicorn 锁屏不生效 行情服务器 股票交易 速度慢 切换 股票量化接口 股票API接口 element-ui 上传视频并预览视频 vue上传本地视频及进度条功能 vue2选择视频上传到服务器 upload上传视频组件插件 批量上传视频 限制单个上传视频 mariadb ecmascript 商用密码产品体系 rsync openEuler 服务网格 istio linux安装配置 换源 Debian 微信分享 Image wxopensdk 混合开发 环境安装 JDK 灵办AI HarmonyOS Next DevEco Studio rclone AList fnOS 历史版本 下载 本地部署AI大模型 产品经理 protobuf 序列化和反序列化 Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 匿名管道 沙盒 openvpn server openvpn配置教程 centos安装openvpn 开发环境 perf linux内核 华为证书 HarmonyOS认证 华为证书考试 matplotlib fonts-noto-cjk lvm iNode Macos gpt-3 Xshell 冯诺依曼体系 pthread 系统 node mybase 代码 对比 meld DiffMerge 创业创新 react native PostgreSQL15数据库 eNSP 企业网络规划 华为eNSP 网络规划 linux上传下载 can 线程池 openwrt USB网络共享 运维监控 minicom 串口调试工具 DocFlow glibc Reactor k8s集群资源管理 云原生开发 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 路径解析 KVM uv anaconda ue4 着色器 ue5 ros2 moveit 机器人运动 telnet 远程登录 ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 etcd RBAC springcloud Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer perl 设置代理 实用教程 读写锁 指令 Typore rancher dubbo 大模型技术 本地部署大模型 Ubuntu22.04 开发人员主页 K8S k8s管理系统 ubuntu安装 linux入门小白 compose Java LInux RDP autoware 智能体 autogen openai coze 源代码 python高级编程 Ansible elk stack lstm LSTM-SVM 时间序列预测 传统数据库升级 银行 嵌入式实习 华为机试 AD域 MS Materials zotero WebDAV 同步失败 代理模式 Kylin OS intellij idea springboot unity3d HCIE 数通 豆瓣 追剧助手 迅雷 多个客户端访问 IO多路复用 TCP相关API 查询数据库服务IP地址 SQL Server PVE 磁盘监控 VPS 外网访问 X11 Xming Unity Dedicated Server Host Client 无头主机 文件分享 MI300x 显示管理器 lightdm gdm 雨云服务器 IPMITOOL 硬件管理 hugo grub 版本升级 扩容 skynet aarch64 编译安装 HPC 田俊楠 搭建个人相关服务器 CPU 主板 电源 网卡 视频平台 录像 视频转发 视频流 硅基流动 ChatBox 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 HP Anyware 视频监控 内网渗透 靶机渗透 智能电视 程序化交易 量化交易 高频交易 券商 股票交易接口api 类型 特点 设计规范 华为昇腾910b3 qwen2vl finebi 教育电商 pyscenic 生信教程 电子信息工程 医疗APP开发 app开发 deekseek 捆绑 链接 谷歌浏览器 youtube google gmail alias unalias 别名 黑苹果 MacOS录屏软件 ArcTS 登录 ArcUI GridItem Ubuntu Server Ubuntu 22.04.5 Linux24.04 Hive环境搭建 hive3环境 Hive远程模式 多层架构 解耦 Redis Desktop Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 Linux的基础指令 VMware安装mocOS macOS系统安装 neo4j safari trea idea ArkTs 进程优先级 调度队列 进程切换 高级IO epoll 磁盘清理 curl wget ubuntu 18.04 UOS 开机自启动 桌面快捷方式 电脑桌面出现linux图标 电脑桌面linux图标删除不了 电脑桌面Liunx图标删不掉 linux图标删不掉 Echarts图表 折线图 柱状图 异步动态数据 鸿蒙开发 可视化效果 brew 材料工程 js jdk11安装 jdk安装 openjdk11 openjdk11安装 离线部署dify fd 文件描述符 深度求索 私域 docker compose 开发 环境配置 ShenTong 中间件 可信计算技术 安全架构 文心一言 序列化反序列化 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos Docker Hub docker pull daemon.json 显卡驱动 增强现实 沉浸式体验 技术实现 案例分析 AR Cookie log4j docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 聚类 mq 嵌入式系统开发 输入法 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 监控k8s 监控kubernetes WLAN DBeaver tcp burp suite 抓包 centos-root /dev/mapper yum clean all df -h / du -sh 免密 公钥 私钥 网易邮箱大师 宝塔面板无法访问 健康医疗 trae 全文检索 图搜索算法 物理地址 页表 虚拟地址 命令模式 桥接模式 windows虚拟机 虚拟机联网 AI提示词优化 企业风控系统 互联网反欺诈 DDoS攻击 SQL注入攻击 恶意软件和病毒攻击 stm32项目 c 网卡的名称修改 eth0 ens33 SSH Xterminal VSCode 物联网开发 webstorm ocr odoo 服务器动作 Server action DeepSeek行业应用 Heroku 网站部署 nextjs reactjs 僵尸世界大战 游戏服务器搭建 HAProxy 银河麒麟桌面操作系统 hibernate vSphere vCenter 软件定义数据中心 sddc 银河麒麟服务器操作系统 系统激活 XFS xfs文件系统损坏 I_O error AI agent dns是什么 如何设置电脑dns dns应该如何设置 ldap 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 nuxt3 服务器部署ai模型 iis Netty 即时通信 NIO DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 Dell R750XS java-rocketmq 远程看看 远程协助 银河麒麟高级服务器 外接硬盘 Kylin 计算虚拟化 弹性裸金属 崖山数据库 YashanDB swoole Java Applet URL操作 服务器建立 Socket编程 网络文件读取 能力提升 面试宝典 技术 IT信息化 致远OA OA服务器 服务器磁盘扩容 网络穿透 netty 加解密 Yakit yaklang JAVA mybatis 剧本 温湿度数据上传到服务器 Arduino HTTP 半虚拟化 硬件虚拟化 Hypervisor 网站搭建 serv00 博客 VMware创建虚拟机 .net mvc断点续传 KylinV10 麒麟操作系统 Vmware 信创 信创终端 中科方德 迁移指南 gradle TrueLicense Jellyfin 安防软件 飞牛 dba 互信 话题通信 服务通信 Web服务器 多线程下载工具 PYTHON windows 服务器安装 samba clickhouse 服务器部署 本地拉取打包 笔灵AI AI工具 海康 vCenter服务器 ESXi主机 监控与管理 故障排除 日志记录 RK3568 百度云 矩池云 数据下载 数据传输 web开发 socket 机器人仿真 模拟仿真 HarmonyOS MateBook 深度优先 数学建模 多端开发 智慧分发 应用生态 鸿蒙OS webgl 数据仓库 数据库开发 database 图形渲染 VMware安装Ubuntu Ubuntu安装k8s 命令行 基础入门 链表 飞牛NAS 飞牛OS MacBook Pro regedit 开机启动 EMUI 回退 降级 embedding 分析解读 prometheus数据采集 prometheus数据模型 prometheus特点 cmos conda配置 conda镜像源 CentOS Stream CentOS 模拟实现 Linux PID Ubuntu共享文件夹 共享目录 Linux共享文件夹 c/c++ 串口 系统开发 binder framework 源码环境 强制清理 强制删除 mac废纸篓 基础环境 x64 SIGSEGV xmm0 iperf3 带宽测试 Linux权限 权限命令 特殊权限 Reactor反应堆 sublime text cmake MVS 海康威视相机 deep learning Tabs组件 TabContent TabBar TabsController 导航页签栏 滚动导航栏 动静态库 推荐算法 视觉检测 实时内核 Ardupilot 一切皆文件 可执行程序 photoshop Github加速 Mac上Github加速 Chrome浏览器插件 切换root stable diffusion nano isaacgym ceph ollama下载加速 jina 跨域 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm AI代码编辑器 金仓数据库 2025 征文 数据库平替用金仓 Open WebUI SSH 密钥生成 SSH 公钥 私钥 生成 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 HiCar CarLife+ CarPlay QT RK3588 DenseNet linux 命令 sed 命令 Linux的权限 springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 vr bat 软负载 监控k8s集群 集群内prometheus Node-Red 编程工具 流编程 WebUI DeepSeek V3 proxy模式 easyui CLion DeepSeek r1 cfssl selete 链路聚合 端口聚合 win11 windows11 NLP模型 NLP edge浏览器 思科 lb 协议 import save load 迁移镜像 ebpf uprobe scikit-learn GameFramework HybridCLR Unity编辑器扩展 自动化工具 qt5 客户端开发 ROS2 蓝牙 大屏端 进程程序替换 execl函数 execv函数 execvp函数 execvpe函数 putenv函数 音乐服务器 Navidrome 音流 rdp 远程服务 论文阅读 postgres Dify重启后重新初始化 solr ranger MySQL8.0 hosts hosts文件管理工具 开源软件 动态库 GCC编译器 -fPIC -shared 超级终端 多任务操作 提高工作效率 dns 电子器件 二极管 三极管 三维重建 实验 远程过程调用 Windows环境 无法解析服务器的名称或地址 SVN Server tortoise svn vasp安装 autodl kylin v10 麒麟 v10 Ubuntu 24.04.1 轻量级服务器 RTMP 应用层 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 流式接口 junit ecm bpm 压测 ECS 游戏机 TrinityCore 魔兽世界 恒源云 Wi-Fi bootstrap 统信操作系统 图论 并集查找 换根法 树上倍增 SSL 域名 Spring Security 移动云 windows日志 ISO镜像作为本地源 MNN Qwen vscode 1.86 ukui 麒麟kylinos openeuler 社交电子 直流充电桩 充电桩 内网环境 token sas Nuxt.js ux llama3 Chatglm 开源大模型 thingsboard cpp-httplib 联想开天P90Z装win10 bonding 思科模拟器 Cisco gateway Clion Nova ResharperC++引擎 Centos7 远程开发 SWAT 配置文件 服务管理 网络共享 执法记录仪 智能安全帽 smarteye SysBench 基准测试 代码调试 ipdb W5500 OLED u8g2 sqlite3 技术共享 矩阵 工业4.0 域名服务 DHCP 符号链接 配置 繁忙 解决办法 替代网站 汇总推荐 AI推理 欧标 OCPP 状态模式 小智AI服务端 xiaozhi TTS iBMC UltraISO 服务器管理 配置教程 网站管理 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 visual studio UOS1070e Qwen2.5-VL 云桌面 微软 AD域控 证书服务器 WebServer powerbi 联机 僵尸毁灭工程 游戏联机 开服 弹性服务器 CNNs 图像分类 动态规划 软件开发 pyside6 界面 Webserver 异步 MinIO Qt QModbus 机架式服务器 1U工控机 国产工控机 计算机系统 shard 源代码管理 OpenGL SoC lua anonymous 热键 HTTP状态码 客户端错误 服务器端错误 API设计 slave 僵尸进程 信号 小艺 Pura X sequoiaDB navicat sonoma 自动更新 deepseek r1 OpenHarmony keepalived Attention threejs 3D mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 代理服务器 交叉编译 Ark-TS语言 HarmonyOS NEXT 原生鸿蒙 支付 微信支付 开放平台 etl minecraft vpn 软件构建 DIFY IO模型 内核 网络文件系统 多产物 Mac软件 AI作画 initramfs Linux内核 Grub 空间 查错 sublime text3 macbook systemctl composer firewalld 管道 接口返回 harmonyosnext SPI nacos TRAE VLAN 企业网络 互联网医院 多路转接 MDK 嵌入式开发工具 kerberos ssh远程登录 IPv4 子网掩码 公网IP 私有IP 代理 OpenSSH P2P HDLC firewall PPI String Cytoscape CytoHubba 虚幻引擎 流水线 脚本式流水线 lio-sam SLAM 实战案例 免费域名 域名解析 OpenManus 接口优化 k8s二次开发 virtualbox copilot 底层实现 ajax ICMP 内存管理 考试 脚本 GRE Alist mount 挂载 网盘 UFW 线程同步 线程互斥 条件变量 Crawlee finalsheel risc-v 多媒体 BitTorrent 搜索 pppoe radius SSL证书 code-server CORS pyqt 飞书 无桌面 黑客技术 SEO iDRAC R720xd jetty undertow 相差8小时 UTC 时间 sysctl.conf vm.nr_hugepages mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 KingBase URL nfs 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 Erlang OTP gen_server 热代码交换 事务语义 tailscale derp derper 中转 业界资讯 xss Unity插件 Anolis nginx安装 linux插件下载 sentinel 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 分布式训练 vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 wsgiref Web 服务器网关接口 LORA XCC Lenovo idm OD机试真题 服务器能耗统计 智能音箱 智能家居 音乐库 西门子PLC 通讯 win服务器架设 windows server less 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? MQTT协议 消息服务器 代码托管服务 云耀服务器 React Next.js 开源框架 Web应用服务器 浪潮信息 AI服务器 代码规范 zerotier 联网 easyconnect 树莓派 显示器 金仓数据库概述 金仓数据库的产品优化提案 日志分析 系统取证 AimRT 支持向量机 pxe ECS服务器 openssh 粘包问题 语法 av1 电视盒子 机顶盒ROM 魔百盒刷机 VR手套 数据手套 动捕手套 动捕数据手套 移动开发 元服务 应用上架 玩机技巧 软件分享 软件图标 合成模型 扩散模型 图像生成 Linux环境 cd 目录切换 用户缓冲区 查看显卡进程 fuser su sudo fork wait waitpid exit ELF加载 csrutil mac恢复模式进入方法 SIP 恢复模式 Maven 用户管理 beautifulsoup 玩游戏 大版本升 升级Ubuntu系统 Office llamafactory 微调 Charles STL cpolar 流程图 mermaid devmem 容器技术 wifi驱动 RockyLinux Bluetooth 配对 进程等待 内存泄漏 figma SSH 服务 SSH Server OpenSSH Server ubuntu24.04.1 fast TCP协议 xpath定位元素 bot 浏览器开发 AI浏览器 openstack Xen seatunnel ubuntu24 vivado24 yum源切换 更换国内yum源 自动化编程 烟花代码 烟花 元旦 端口 查看 ss 项目部署 网络建设与运维 网络搭建 神州数码 神州数码云平台 云平台 显示过滤器 Wireshark安装 问题解决 钉钉 自学笔记 小米 澎湃OS Android 解决方案 System V共享内存 进程通信 Sealos 电视剧收视率分析与可视化平台 图片增强 增强数据 Mermaid 可视化图表 Zoertier 内网组网 Docker快速入门 netlink libnl3 GeneCards OMIM TTD VGG网络 卷积层 池化层 WireGuard 异地组网 ufw SystemV 计算机科学与技术 sudo原理 su切换 IPv6 IPv6测试 IPv6测速 IPv6检测 IPv6查询 kubeless client-go Linux的基础开发工具 Trae叒更新了? VM虚拟机 anythingllm open-webui docker国内镜像 MLLMs VLM gpt-4v 文件共享 openresty 工作流自动化工具 bushujiaocheng 部署教程 AI算力 租算力 到算家云 laravel 根服务器 wpf 服务器时间 es h.264 聊天室 sqlite 游戏开发 干货分享 黑客工具 密码爆破 架构与原理 chfs ubuntu 16.04 服务器主板 AI芯片 IDEA 北亚数据恢复 oracle数据恢复 VNC VS Code AD 域管理 wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 超融合 端口测试 saltstack nosql 数字证书 签署证书 banner client close Carla 授时服务 北斗授时 服务器ssl异常解决 glm4 deepseak 豆包 KIMI 腾讯元宝 高效I/O VPN wireguard 网络原理 umeditor粘贴word ueditor粘贴word ueditor复制word ueditor上传word图片 ueditor导入word ueditor导入pdf ueditor导入ppt logstash AWS 火山引擎 bigdata fpga sse_starlette Starlette Server-Sent Eve 服务器推送事件 移动魔百盒 ArtTS whistle 搜狗输入法 中文输入法 跨平台 xfce 终端 archlinux kde plasma retry 重试机制 子系统 NVM Node Yarn PM2 提示词 Bandizip Mac解压 Mac压缩 压缩菜单 互联网实用编程指南 影视app RAGflow mac安装软件 mac卸载软件 mac book clipboard 剪贴板 剪贴板增强 mac cocoapods macos cocoapods Mac部署 Ollama模型 Openwebui 配置教程 AI模型 开发工具 容器清理 大文件清理 空间清理 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 docker命令大全 docker部署Python 李心怡 docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 抗锯齿 dock 加速 deployment daemonset statefulset cronjob 集群管理 Helm k8s集群 生活 静态IP Masshunter 质谱采集分析软件 使用教程 科研软件 fabric podman registries HBase分布式集群 HBase环境搭建 HBase安装 HBase完全分布式环境 Radius 线性代数 电商平台 国标28181 监控接入 语音广播 流程 SDP 佛山戴尔服务器维修 佛山三水服务器维修 宕机切换 服务器宕机 AISphereButler flink 流量运营 ardunio BLE hexo 服务器正确解析请求体 配置原理 负载测试 yashandb 充电桩平台 充电桩开源平台 solidworks安装 免费 massa sui aptos sei java-zookeeper paddle lrzsz Putty 花生壳 cocos2d 3dcoat PTrade QMT 量化股票 概率论 accept 自定义登录信息展示 motd 美化登录 h.265 腾讯云服务器 轻量应用服务器 linux系统入门 linux命令 issue linq 站群服务器 性能监控 风扇控制软件 极限编程 pyicu 星河版 NVIDIA win向maOS迁移数据 欧拉系统 mvc java毕业设计 微信小程序医院预约挂号 医院预约 医院预约挂号 小程序挂号 miniconda 顽固图标 启动台 ubantu GKI KMI IP地址 计算机基础 苹果电脑装windows系统 mac安装windows系统 mac装双系统 macbook安装win10双 mac安装win10双系统 苹果电脑上安装双系统 mac air安装win 系统内核 Linux版本 树莓派项目 红黑树 uni-app x 文件清理 tftp 达梦 DM8 AzureDataStudio 机械臂 webview docker search d3d12 知行EDI 电子数据交换 知行之桥 EDI homeassistant 医院门诊管理系统 轮播图 BCLinux GPUGEEK 代理配置 企业级DevOps libreoffice 转换 恒玄BES rxjava 三次握手 caddy 实时云渲染 云渲染 3D推流 静态NAT 零日漏洞 CVE OpenAI 独立服务器 macOS 数据库管理 qtcreator AppLinking 应用间跳转 自定义shell当中管道的实现 匿名和命名管道 function address 函数 地址 进程池实现 direct12 C/C++ Linux指令 时间轮 泰山派 根文件系统 编译 烧录 嵌入式实时数据库 linux常用命令 模块测试 NVIDIA显卡安装 Ubuntu开机黑屏 coffeescript 云盘 安全组 零售 输入系统 watchtower 智能问答 Spring AI Milvus 红黑树封装map和set web环境 代码复审 LLaMA-Factory 集成 scrapy 通用环境搭建 MySQL 5分钟快速学 docker入门 dockerfile 共享 设置 nacos容器环境变量 docker启动nacos参数 nacos镜像下载 Linux系统编程 根目录 WINCC OpenManage LVS rtcp CPU架构 服务器cpu vue在线预览excel和编辑 vue2打开解析xls电子表格 浏览器新开页签或弹框内加载预览 文件url地址或接口二进制文档 解决网页打不开白屏报错问题 光电器件 LED 大厂程序员 硅基计算 碳基计算 认知计算 生物计算 AGI 系统架构设计 软件哲学 程序员实现财富自由 5090 显卡 AI性能 A2A Svelte 全栈 机架式 IDC aiohttp asyncio raid bug定位 缺陷管理 雾锁王国 Metastore Catalog GRANT REVOKE 导航栏 nginx默认共享目录 青少年编程 broadcom 基础指令 本地不受DeepSeek word转pdf charles chrome历史版本下载 chrominum下载 路径规划 环境部署 文档 Autoware 辅助驾驶 CAD瓦片化 栅格瓦片 矢量瓦片 Web可视化 DWG解析 金字塔模型 mapreduce 定义 核心特点 优缺点 适用场景 Multi-Agent 蜂窝网络 频率复用 射频单元 无线协议接口RAN 主同步信号PSS 4 - 分布式通信、分布式张量 烟雾检测 yolo检测 消防检测 springboot容器部署 springboot容器化部署 微服务容器化负载均衡配置 微服务容器多节点部署 微服务多节点部署配置负载均衡 tar.gz tar.xz linux压缩 高可用 仓库 影刀证书 分享 服务注册与发现 电脑操作 CodeBuddy首席试玩官 dify部署 生成对抗网络 STP 生成树协议 PVST RSTP MSTP 防环路 网络基础 proxy_pass Playwright MCP 电子学会 Nginx报错413 Request Entity Too Large 的客户端请求体限制 authing aac openvino 环境 非root 能源 access blocked 破解 动态域名 惠普服务器 惠普ML310e Gen8 惠普ML310e Gen8V2 站群 多IP 风扇散热策略 曙光 海光 宁畅 中科可控 RNG 状态 可复现性 随机数生成 事件分析 边缘服务器 利旧 AI识别 分布式数据库 集中式数据库 业务需求 选型误 客户端-服务器架构 点对点网络 服务协议 网络虚拟化 网络安全防御 科勘海洋 数据采集浮标 浮标数据采集模块 Cache Aside Read/Write Write Behind 分类 AI导航站 视频服务器 proto actor actor model Actor 模型 bert swift 软件商店 livecd systemtools Ubuntu 24.04 搜狗输入法闪屏 Ubuntu中文输入法 实时日志 logs NTP服务器 能效分析 端口开放 pipe函数 管道的大小 匿名管道的四种情况 ps命令 SFTP 手动分区 cuda驱动 Navigation 路由跳转 鸿蒙官方推荐方式 鸿蒙原生开发 Python学习 Python编程 FS bootfs rootfs linux目录 Linux的进程控制 亲测 linux/cmake GenAI LLM 推理优化 LLM serving tvm安装 深度学习编译器 homebrew windows转mac ssh密匙 Mac配brew环境变量 gstreamer 冯诺依曼体系结构 Eigen zephyr lvgl8.3 lvgl9.2 lvgl lvgl安装 手机 Modbus TCP 容器化 Serverless 基本指令 模板 containerd 访问公司内网 软路由 线程安全 微信自动化工具 微信消息定时发送 文件权限 CAN 多总线 docker安装mysql win下载mysql镜像 mysql基本操作 docker登陆私仓 docker容器 deepseek与mysql 多系统 仙盟大衍灵机 东方仙盟 仙盟创梦IDE 触觉传感器 GelSight GelSightMini GelSight触觉传感器 dnf Quixel Fab UE5 游戏商城 electron pnet 百度 pnetlab 工厂方法模式 vmware tools 制造 流量 证书 签名 排序算法 选择排序 rtmp gitlab服务器 物理服务器 服务器租用 云服务器租用 物理机租用 网络接口 时间间隔 所有接口 多网口 DELL R730XD维修 全国服务器故障维修 网络IO 队列 数据库占用空间 eventfd 高性能 Modbustcp服务器 SFTP服务端 js逆向 OS orbslam2 ANDROID_HOME zshrc latex 更换镜像源 IT 护眼模式 Linux的进程概念 重置密码 codereview code-review 系统完整性 越狱设备 Windows应急响应 应急响应 webshell 网络攻击防御 网络攻击 VUE Mysql PATH 命令行参数 main的三个参数 激光雷达 镭眸 机床 仿真 课件 教学 课程 九天画芯 铁电液晶 显示技术 液晶产业 技术超越 数码相机 全景相机 设备选择 实用技巧 数字空间 二级页表 杂质 elementui 若依框架 webgis cesium 视频号 HP打印机 RHCE 数据库数据恢复 漏洞报告生成 学习路线 Web3 Telegram nmcli 网络配置 安全性测试 写时拷贝 Linux的进程调度队列 活动队列 sql注入 回归 原子操作 AXI MinerU 权限掩码 粘滞位 量子计算 muduo库 gerrit usb typec 自动化测试框架 Typescript Unlocker 编译器 计算机八股 huggingface N8N CKA docker 失效 docker pull失效 docker search超时 AI控制浏览器 Browser user 集合 List 开发效率 Windmill vb SonarQube 泛型编程 Async注解 事件驱动 janus MCP 服务器 JADX-AI 插件 jQuery EF Core 客户端与服务器评估 查询优化 数据传输对象 查询对象模式 实时传输 服务器托管 云托管 数据中心 idc机房 linux cpu负载异常 LangGraph 模型上下文协议 MultiServerMCPC load_mcp_tools load_mcp_prompt 若依 内存不足 outofmemory Key exchange 主包过大 Windows 11 重装电脑系统 鼠标 teamspeak 磁盘IO iostat pikachu靶场 XSS漏洞 XSS DOM型XSS FreeRTOS 进程管理 麒麟OS Lenovo System X GNOME Scoket labview zipkin flinkcdc 观察者模式 SQI iOS Trust Authentication Challenge mujoco 迁移 C++11 lambda 命令键 Makefile Windows Hello 摄像头 指纹 生物识别 bpf bpfjit pcap vr看房 在线看房系统 房产营销 房产经济 三维空间 汽车 软硬链接 文件 药品管理 HarmonyOS SDK Map Kit 地图 桶装水小程序 在线下单送水小程序源码 桶装水送货上门小程序 送水小程序 订水线上商城 English 服务 鸿蒙项目 程序地址空间 云解析 云CDN SLS日志服务 云监控 #Linux #shell #脚本 vsode arkts arkui dnn Make 单用户模式 HarmonyOS 5开发环境 智能手表 Pura80 WATCH 5 nvcc A100 数字比特流 模拟信号 将二进制数据映射到模拟波形上 频谱资源 振幅频率相位 载波高频正弦波 MobileNetV3 深度强化学习 深度Q网络 Q_Learning 经验回收 Jenkins流水线 声明式流水线 笔记本电脑 dsp开发 pow 指数函数 优化 AOD-PONO-Net 图像去雾技术 NAT 软件安装 Agentic Web NLWeb 自然语言网络 微软build大会 编程与数学 ROS1/ROS2 Wayland VAD 视频异常检测 VAR 视频异常推理 推理数据集 强化微调 GRPO 开启关闭防火墙 BMS 储能 机器人操作系统 责任链模式 高考 省份 年份 分数线 数据 authorized_keys 密钥 改行学it openlayers bmap tile server TraeAgent Spring Boot 进程状态 http状态码 请求协议 asp.net 低成本 网络带宽 问题排查 RHEL 去中心化 过期连接 相机标定 Featurize Mobilenet 分割 信创国产化 达梦数据库 CLI JavaScript langgraph.json messages dmesg 报警主机 豪恩 VISTA120 乐可利 霍尼韦尔 枫叶 时刻 矩阵乘法 3D深度学习 udp回显服务器 CSDN开发云 大学大模型可视化教学 全球气象可视化 大学气象可视化 funasr asr 语音转文字 算法协商 故障排查 Web测试 gpu 大模型应用开发 AI 应用商业化 参数服务器 分布式计算 数据并行 加密 哥sika DICOM CTE AGE debezium 数据变更 android-ndk illustrator octomap_server react Native 学习笔记 包装类 CKEditor5 源码软件 调试方法 Valgrind 内存分析工具 实战项目 入门 精通 CTF 时序数据库 iotdb 路由器 CUDA Toolkit 恢复 对话框showDialog showActionMenu 操作列表ActionSheet CustomDialog 文本滑动选择器弹窗 消息提示框 警告弹窗 libtorch rk3588 rknn-toolkit2 Arduino下载开发板 esp32开发板 esp32-s3 记账软件 容器部署 ICMPv6 network NetworkManager unionFS OverlayFS OCI docker架构 写时复制 简单工厂模式 客户端和服务器端 pandas Cilium 最新微服务 思科实验 高级网络互联 中文分词 麒麟kos 网络检测 ping UDS Bootloader 嵌入式软件 guava 实时语音识别 流式语音识别 BiSheng 驱动器映射 批量映射 win32wnet模块 网络驱动器映射工具 shell编程 地平线5 OSB Oracle中间件 SOA 服务发现 nvidia驱动 Tesla显卡 MQTT Broker GMQT 客户端/服务器架构 分布式应用 三层架构 Web应用 跨平台兼容性 dfs blender three.js 数字孪生 决策树 漫展 滑动验证码 反爬虫 英语六级 cs144 接口隔离原则 scala cangjie struts 物联网嵌入式开发实训室 物联网实训室 嵌入式开发实训室 物联网应用技术专业实训室 物理机 Java 日志框架 Log4j2 Logback SLF4J 结构化日志 企业级应用 材质 贴图 BIO Java socket Java BIO Java NIO Java 网络编程 alphafold3 gemini gemini国内访问 gemini api gemini中转搭建 Cloudflare ESP8266简单API服务器 Arduino JSON webserver 支付宝小程序 云开发 FS100P pve TCP回显服务器 jvm调优 LRU策略 内存增长 垃圾回收 分布式总线 美食 更新apt 安装hadoop前的准备工作 FreeLearning EulerOS 版本对应 Linux 发行版 企业级操作系统 开源社区 进度条 FCN WebFuture OpenTiny 打包工具 WinCC OT与IT SCADA 智能制造 MES 信奥 dataworks maxcompute 几何绘图 三角函数 SSM 项目实战 页面放行 物理层 重构 迭代器模式 数字化转型 局域网 数据链路层 开启黑屏 tty2 抽象工厂模式 医药 协作 快速入门 cn2 带宽 mcp client mcp server 弹性 mysql 8 mysql 8 忘记密码 RustDesk 搭建服务器 AudioLM 华为OD机考 机考真题 需要广播的服务器数量 mobaxterm RTX5090 torch2.7.0 pavucontrol 蓝牙耳机 dockercompose安装 compose.yml文件详解 dockercompose使用 cordova 跨域开发 RTOS time时间函数