- 替换原理
用fork创建子进程后执行的是和父进程相同的程序,也有可能执行不同的代码分支,子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新进程替换,从新程序的启动例程(main函数)开始执行。
注意:调用exec并不创建新进程,所以调用exec前后该进程的id并为改变。exec只是用磁盘上的一个新程序替换了当前进程的正文段、数据段、堆段和栈段。
- 替换函数
有六种以exec开头的函数,统称exec函数:
#include <unistd.h>
int execl(const char *path,char *argv0,char*argv1......(char *)0);
int execv(const char *path,char *argv[]);
int execlp(const char *file,char *argv0,char*argv1......(char *)0);
int execvp(const char *file,char *argv[]);
int execle(const char *path,char *argv0,char*argv1......(char *)0,char *envp[]);
int execve(char *path,char *argv[],char *envp[]);
- 返回值:这些函数如果调用成功,则加载新程序从启动代码开始执行,不再返回。如果调用出错,则返回-1。可知,exec函数只有出错的返回值,而没有成功的返回值。
- 参数表的传递:
1)函数execl、execle、execlp要求将新程序的每个命令行参数都说明为一个单独的参数,这种参数表以空指针结尾。
2)对于另外3个函数,则先构造一个指向各参数的指针数组,然后将该数组地址作为这3个函数的参数。
事实上,只有execve 是真正的系统调用,其他五个函数最终都调用 execve。