Linux进程控制–exec家族

Share To Facebook Twitter

用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等同。用参数传递取代了重定向。

(责任编辑:IT)

Posted in 未分类

发表评论