进程替换
我们之前说fork()创建子进程,是为了让子进程执行和父进程相同的程序,并且父进程只能获得子进程的一些特殊的信息,而如果我们需要子进程执行一些和父进程不同的程序,这里就需要进程替换了。
父进程创建子进程后,子进程调用exec函数来执行另一个程序,
进程的虚拟地址、物理地址以及硬盘之间映射关如下图:
那进程是如何替换的呢?
进程将当前要替换的程序的数据和代码将原有的数据和代码覆盖掉,从新程序的启动例程开始执行,调用exec并不创建新进程,所以调用exec函数前后i该进程id并不改变,
替换函数
int execl(const char* path,const char* arg,…)
带路径和操作选项
execl("/bin/ps", "ps", "-ef", NULL);
int execlp(cosnt char* file,const char* arg,…)
带文件名和操作选项
execlp("ls","-a","-i","-o",NULL);
int execv(const char *path, char *const argv[])
带路径和操作选项数组
execv("/bin/ps",argv);
int execle(const char *path, const char *arg, …,char *const envp[])
带路径和操作选项以及自己配的环境变量
execle("./bin/echoenv","echoenv",NULL,env_init);
int execvp(const char *file, char *const argv[])
带文件名和操作选项数组
execvp("ps",argv);
int execve(const char *path, char *const argv[], char *const envp[])
带路径和操作选项数组,环境变量
execve("/bin/ps", argv, envp);
exec函数族的函数执行成功后不会返回,调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。
实上,只有execve是真正的系统调用,其它五个函数最终都调用 execve,所以execve在man手册 第2节,其它函数在man手册第3节。这些函数之间的关系如下图所示。
下图exec函数族 一个完整的例子: