Linux进程控制--exec家族
原创
©著作权归作者所有:来自51CTO博客作者luteresa的原创作品,请联系作者获取转载授权,否则将追究法律责任
用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并未改变
#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[]);
这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回,如果调用出错则返回-1,所以exec函数只有出错的返回值而没有成功的返回值。
exec函数记忆方法:
1.exec后第一部分,是l表示,新执行的进程参数以列表方式传入,若是v,则以指针数组传入,数组中的最后一个指针也应该是NULL,就像main函数的argv参数或者环境变量表一样。
2.exec第二部分,带p(path),在PATH环境变量的目录列表中搜索这个程序。不带p,必须明确给出相对路径,或绝对路径名。对于以e(表示environment)结尾的exec函数,可以把一份新的环境变量表传给它,其他exec函数仍使用当前的环境变量表执行新程序。
事实上,只有execve是真正的系统调用,其它五个函数最终都调用execve,所以execve在man手册第2节,其它函数在man手册第3节。
看个实例
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
int main(int argc,char *argv[])
{
int fd;
if (argc!=2){
fputs("usage:wrapper file\n",stderr);
exit(1);
}
fd = open(argv[1],O_RDONLY);
if (fd<0){
perror("open");
exit(1);
}
dup2(fd,STDIN_FILENO);
close(fd);
execl("./upper","upper",NULL);
perror("exec ./upper");
exit(1);
}
这个程序执行动作是:
./wrapper file.txt 实际效果是与
./upper < file.txt等同。用参数传递取代了重定向。