【Linux】进程替换

img

Halo,这里是Ppeua。平时主要更新C语言,C++,数据结构算法…感兴趣就关注我吧!你定不会失望。


在这里插入图片描述# 0. 进程替换概念

我们想要在一个进程中的子进程运行外部程序,就可以用到进程替换的相关接口.

系统对于此提供了exec类的相关接口.

相关接口有6个,不过都具有一定的使用规则.

image-20231116081306274

先来看看单进程版的进程替换

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
int main()
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    execl("/usr/bin/ls","ls","-a","-l",NULL);
    printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    return 0;
}

编译运行会发生什么呢?

image-20231116081848443

执行到execl前的程序是正常运行的,也就是正常打印出了进程PID相关信息.

后来执行到了execl,如我们猜测的一样执行了 ls -al 的指令

但是!!后来的 **printf(“after i am a process,pid:%d,ppid:%d\n”,getpid(),getppid()); **这条指令并未执行.

这是我们没有想到的?这是为什么呢?

回顾我们的标题名,叫做进程替换.显然这里发生了进程的替换.

8df75dddf104585179d4c2eed3338dd

一个文件未被运行时其在磁盘当中,execl把内存当中的ls可执行程序,对内存中的./proce可执行程序进行了一个替换.不创建新的进程,只进行代码和数据的替换

注意!这里并未重新创建一个进程,而是在原有的进程上进行了夺舍,也就是运行了exec*指令后,进程还是那个进程,内容却不再是原来的内容了(肉体仍旧,灵魂改变).

所以替换成功后,整个程序改变了,那么这个函数有返回值嘛

运行成功后数据替换,就算返回了也被换掉,所以执行成功是没有返回值的,执行失败则返回-1

现在我们大概能知道什么是进程替换了.我们在看看在多进程状态下,进程替换会发生什么呢?

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
int main(int argc,char *argv[])
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    pid_t ret=fork();
    
    if(ret == 0)
    {
        printf("before i am a child process,pid:%d,ppid:%d\n",getpid(),getppid());   
        execl("/usr/bin/ls","ls","-a","-l",NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    
    else
    {
        int ret=wait(NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    return 0;
}

image-20231116084200236

子进程执行了execl指令,完成了进程替换,而父进程照常运行自己剩下的指令.

那么,这几个接口有什么含义呢.以及那个命令为什么要这样写呢?

1. 进程替换库函数接口

exec*相关接口使用需包含两个库

#include<sys/types.h>
#include<sys/wait.h>

其中函数的命名是有一定的规律的

  1. l:list
  2. p:path
  3. v:vector
  4. e:environ

execl与execv

观察六个函数我们发现,前缀要么是 execl,要么是 execv,这具体含义是

l: list 列表 v: vector 数组

也就是

execl("可执行文件所在路径","文件运行时指令","NULL")

文件运行指令在屏幕上输入什么,就在这里输入了什么

  • 例如: ls -al 我们在屏幕上需要输入这两个指令+参数.所以而ls所在路径为 /usr/bin/ls,最后以NULL结尾

    execl("/usr/bin/ls","ls","-al",NULL);
    

同样也可以运行我们自己的可执行程序 并不仅限于C,可以是任何可执行程序!

test.c:

#include<stdio.h>
int main()
{
    printf("hello\n");
    return 0;
}

proce.c:

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
int main(int argc,char *argv[])
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    pid_t ret=fork();
    
    if(ret == 0)
    {
        printf("before i am a child process,pid:%d,ppid:%d\n",getpid(),getppid());   
        execl("./test","test",NULL); //可以为相对路径也可以为绝对路径
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    
    else
    {
        int ret=wait(NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    return 0;
}

image-20231116085803827

如何用makefile同时编译多文件

test:test.c
	gcc -o $@ $^ -std=c99
.PHONY:clean
clean:
	rm -rf proce test

通常我们编译单文件是这样.

Makefile会自顶向下执行检测到的第一个依赖项,然后完成编译,不会编译其他依赖项

也就是如果想要这样编译多文件是没有用的

proce:proce.c
	gcc -o $@ $^ -std=c99
test:test.c
	gcc -o $@ $^ -std=c99
.PHONY:clean
clean:
	rm -rf proce test

我们需要添加一个伪目标:

.PHONY:all
all:proce test

proce:proce.c
	gcc -o $@ $^ -std=c99
test:test.c
	gcc -o $@ $^ -std=c99
.PHONY:clean
clean:
	rm -rf proce test

回到上面,我们看过了L如何使用,那么V呢?

其实这两个差不多,我们需要手动创建一个字符数组,将上面的指令放到数组中即可.最后同样需要以NULL结尾

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
int main(int argc,char *argv[])
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    pid_t ret=fork();
    char * myargv[]=
    {
        "ls",
        "-a",
        "-l",
        NULL
    };
    if(ret == 0)
    {
        printf("before i am a child process,pid:%d,ppid:%d\n",getpid(),getppid());   
        // execl("./test","test",NULL);
        execv("/usr/bin/ls",myargv);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    
    else
    {
        int ret=wait(NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    return 0;
}

image-20231116090741443

execlp与execvp

这个P的概念是,系统会从默认的**$PATH**中去寻找可执行程序,不需要再写路径.

这里以execlp举例.

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
int main(int argc,char *argv[])
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    pid_t ret=fork();
    if(ret == 0)
    {
        printf("before i am a child process,pid:%d,ppid:%d\n",getpid(),getppid());   
        execlp("ls","ls","-a","-l",NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    else
    {
        int ret=wait(NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    return 0;
}

**execlp(“ls”,“ls”,“-a”,“-l”,NULL)**这里的第一个ls表示要在PATH中寻找哪个执行程序,后面依然是参数列表.

execle与execvpe

多了一个e的选项:表示可以传递的环境变量列表.

我们用自己创建的程序来验证一下这个信息.

#include<stdio.h>
#include<stdlib.h>
int main(int argc,char * argv[],char * environ[])
{
    for(int i=0;environ[i];i++)
    {
        printf("i -> %d :,environ: %s\n",i,environ[i]);
    }
    printf("end\n");
    return 0;
}
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
int main(int argc,char *argv[])
{
    printf("before i am a process,pid:%d,ppid:%d\n",getpid(),getppid());   
    pid_t ret=fork();
    char *environ[]={
        "hello=123",
        "Linux",
        "execle",
        NULL
    };
    if(ret == 0)
    {
        printf("before i am a child process,pid:%d,ppid:%d\n",getpid(),getppid());   
        execle("./test","test",NULL,environ);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    else
    {
        int ret=wait(NULL);
        printf("after i am a process,pid:%d,ppid:%d\n",getpid(),getppid());
    }
    return 0;
}

image-20231116092051565

现在我们有三种形式向子进程中传入环境变量

  1. 不传入,直接使用全局的环境变量
  2. 在父进程中使用putenv(),追加形式向子进程传入环境变量
  3. 使用自定义的环境变量传参

2. 进程替换系统调用接口

image-20231116125255627

上方提到的六个库函数都是调用此函数

image-20230905164632777

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ppeua

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值