进程程序替换
fork() 之后,⽗⼦各⾃执⾏⽗进程代码的⼀部分如果⼦进程就想执⾏⼀个全新的程序呢?进程的程序
替换来完成这个功能!
程序替换是通过特定的接⼝,加载磁盘上的⼀个全新的程序(代码和数据),加载到调⽤进程的地址空间中!
4.1 替换原理
⽤fork创建⼦进程后执⾏的是和⽗进程相同的程序(但有可能执⾏不同的代码分⽀),⼦进程往往要调⽤⼀种exec函数以执⾏另⼀个程序。当进程调⽤⼀种exec函数时,该进程的⽤⼾空间代码和数据完全被新程序替换,从新程序的启动例程开始执⾏。调⽤exec并不创建新进程,所以调⽤exec前后该进程的id并未改变。
4.2 替换函数
man exec
#include <unistd.h>int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ...,char *const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);
4.2.1 函数解释
- 这些函数如果调⽤成功则加载新的程序从启动代码开始执⾏,不再返回。
- 如果调⽤出错则返回-1
- 所以exec函数只有出错的返回值⽽没有成功的返回值
4.2.1 函数讲解
第一种:
int execl(const char *path, const char *arg, ...);
知识点:对于接下来讲的exec序列函数中的知识点是通用的
1.一旦exec函数替换,就去执行新代码了,原始后面的代码已经不存在了,
2.不做返回值判断,只要返回就是失败了。
引出问题:除了替换指令可以替换自己写的程序吗
先随意写一个语言的代码 c++为例
怎么证明当前你在进行替换的时候,并没有创建新的进程呢,是同一个pid吗?
原理:没有在原替换过程,创建新程序,只是把当前进程的代码和数据用新的程序的代码与数据覆盖式的替换,此时子进程与父进程再也没有关系了,他们分离了。
第二种
execlp("ls","ls","-ln","-a",NULL);
第三种
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main()
{printf("我的程序要运行了!\n");if(fork() == 0){printf("I am Child, My Pid Is: %d\n", getpid());sleep(1);char *const argv[] = { //命令行参数表(char*const)"ls",(char*const)"-l",(char*const)"-a",NULL};execv("/usr/bin/ls", argv);exit(1);}waitpid(-1, NULL, 0);printf("我的程序运行完毕了\n");//return 0;
}
小知识:所有进程所需要的命令行参数都是通过父进程用exec 传的命令行参数
第四种带有环境变量
相关问题
1. 关于 execve
更新环境变量后以前环境变量消失
execve
函数执行成功后,会用新程序完全替换当前进程的执行映像,包括代码段、数据段等。新程序启动时,使用的是通过 envp
参数传入的环境变量数组。如果传入的 envp
中不包含之前进程的某些环境变量,那么在新程序中这些之前的环境变量就不存在了。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main() {char *const argv[] = {"echo", "Hello"};char *const new_envp[] = {"MY_NEW_VAR=value",NULL};// 执行echo程序,传递新的环境变量数组,原进程环境变量未传入if (execve("/bin/echo", argv, new_envp) == -1) {perror("execve");exit(EXIT_FAILURE);}return 0;
}
在上述代码中,execve
执行 echo
程序时,只传入了自定义的环境变量数组 new_envp
,原进程的环境变量没有包含在内,所以 echo
程序运行时看不到原进程的环境变量,就好像之前的环境变量 “消失” 了。
2. 使用 putenv
更新环境变量示例
#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main() {// 定义要设置的环境变量字符串char var_value[] = "MY_VAR=HelloWorld";// 使用putenv设置环境变量if (putenv(var_value) != 0) {perror("putenv");return EXIT_FAILURE;}// 获取并打印环境变量char *value = getenv("MY_VAR");if (value != NULL) {printf("MY_VAR value: %s\n", value);} else {printf("Failed to get MY_VAR\n");}return EXIT_SUCCESS;
}
在这个例子中,putenv
函数用于将 MY_VAR=HelloWorld
这个环境变量添加到当前进程的环境变量表中。putenv
函数接受一个形如 "变量名=变量值"
的字符串参数,如果设置成功,它不会返回错误值;若失败(比如内存分配问题等 ),则返回非零值。之后通过 getenv
函数获取并打印刚刚设置的环境变量的值。
3. 证明不更新环境变量子进程本身有
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main() {pid_t pid = fork();if (pid < 0) {perror("fork");return EXIT_FAILURE;} else if (pid == 0) {// 子进程char *value = getenv("PATH");if (value != NULL) {printf("子进程中PATH环境变量值: %s\n", value);} else {printf("子进程中获取PATH环境变量失败\n");}return EXIT_SUCCESS;} else {int status;pid_t wpid = waitpid(pid, &status, 0);if (wpid == -1) {perror("waitpid");return EXIT_FAILURE;}return EXIT_SUCCESS;}
}
在这段代码中,通过 fork
创建子进程。子进程中使用 getenv
函数获取 PATH
环境变量(没有对环境变量进行更新操作 )。由于子进程会继承父进程的环境变量,所以在子进程中可以获取到父进程传递下来的 PATH
环境变量的值,从而证明子进程本身在不更新环境变量的情况下,是有从父进程继承来的环境变量的。
小知识:其实替换函数有七个 ,上面举例了六个,剩下的哪一个是系统调用的,也就是说,为什么那六个就算不写环境变量,也会本身自带,正是因为那六个是语言封装的,自动会调用系统自带的。