Linux-exec族

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);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值