Linux进程之exec族
exec替换进程映像
Unix在创建进程时采用了一种独特的方法,它将进程创建与加载一个新进程映像分离。这样的好处是有更多的余地对两种操作进行管理。
用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),通常我们将子进程替换成新的进程映像,这时就可以用exec族的函数进行。当进程调用exec函数时,该进程的用户空间和数据完全被新程序替换掉,从新程序的启动例程开始执行。调用exec并不创建新的进程,所以调用exec前后该进程的PID并未改变。
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[]);
exec系列函数只有出错的返回值没有成功的返回值,因为如果exec函数调用成功则加载新的程序从启动代码开始执行,不再返回,如果调用出错才会返回-1。
exec族函数规律
l:可变参数列表
p:在path环境变量中搜素file文件。
例如”/bin/ls”或”./a.out”而不能是”ls”或”a.out”.对于带字母p的函数:
- 如果参数中包含/,则将其视为路径名。
- 否则视为不带路径的程序名,在PATH环境变量的目录列表中搜索这个程序。
v:表示vector,使用命令行参数数组
该函数需要构建一个指向各个参数的指针数组,然后将该数组的首地址作为参数传递给它,数组中的最后一个指针也应该指向NULL。
e:表示environment,使用环境变量数组,不使用进程原有的环境变量,设置新加载程序运行的环境变量
可以把一份新的环境变量表传给它,其
他exec函数仍使用当前的环境变量表执行新程序
exec调用举例如下:
char *const ps_argv[] ={"ps", "-o", "pid,ppid,pgrp,session,tpgid,comm", NULL};
char *const ps_envp[] ={"PATH=/bin:/usr/bin", "TERM=console", NULL};
execl("/bin/ps", "ps", "-o", "pid,ppid,pgrp,session,tpgid,comm", NULL);
execv("/bin/ps", ps_argv);
execle("/bin/ps", "ps", "-o", "pid,ppid,pgrp,session,tpgid,comm", NULL, ps_envp);
execve("/bin/ps", ps_argv, ps_envp);
execlp("ps", "ps", "-o", "pid,ppid,pgrp,session,tpgid,comm", NULL);
execvp("ps", ps_argv);
例子
int main(void)
{
printf("man pid:%d\n",getpid());
int res = execlp("./test",NULL,NULL);
if(res == -1)
{
perror("exec failed");
}
printf("fork after ...\n");
return 0;
}
例子:
#include <unistd.h>
#include <stdlib.h>
int main(void)
{
execlp("ps", "ps", "-o", "pid,ppid,pgrp,session,tpgid,comm", NULL);
perror("exec ps");
exit(1);
}
由于exec函数只有错误返回值,只要返回了一定是出错了,所以不需要判断它的返回值,直接在后面调用perror即可。注意在调用execlp时传了两个“ps”参数,第一个“ps”是程序名,execlp函数要在PATH环境变量中找到这个程序并执行它,而第二个“ps”是第一个命令行参数,execlp函数并不关心它的值,只是简单地把它传给ps程序,ps程序可以通过main函数的argv[0]取到这个参数。
调用exec后,原来打开的文件描述符仍然是打开的。利用这一点可以实现I/O重定向。
先看一个简单的例子,把标准输入转成大写然后打印到标准输出:
例 upper
/* upper.c */
#include <stdio.h>
int main(void)
{
int ch;
while((ch = getchar()) != EOF) {
putchar(toupper(ch));
}
return 0;
}
例 wrapper
/* wrapper.c */
#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);
}