exec系列函数(execl、execlp、execle、execv、execvp)使用

一、exec 替换进程映像

在进程的创建上 Unix 采用了一个独特的方法,它将进程创建与加载一个新进程映象二者分离。这样的好处是有更多的余地对两种操作进行管理。

当我们创建了一个进程之后,通常将子进程替换成新的进程映象,这可以用 exec 系列的函数来进行。当然,exec 系列的函数也可以将当前进程替换掉。

例如:在 shell 命令行执行 ps 命令,实际上是 shell 进程调用 fork 复制一个新的子进程,在利用exec 系统调用将新产生的子进程完全替换成 ps 进程。

二、exec 系列函数(execl、execlp、execle、execv、execvp)

功能:

        用 exec 函数可以把当前进程替换为一个新进程,且新进程与原进程有相同的 PID。exec 名下是由多个关联函数组成的一个完整系列。

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[]);

path 参数表示你要启动程序的名称包括路径名。

arg 参数表示启动程序所带的参数,一般第一个参数为要执行命令名,不是带路径且 arg 必须以NULL 结束。

返回值:成功返回 0,失败返回 -1。

注:上述 exec 系列函数底层都是通过 execve 系统调用实现。

#include <unistd.h>

int execve(const char *filename, char *const argv[],char *const envp[]);

DESCRIPTION:
       execve() executes the program pointed to by filename.  
       filename must be either a binary executable, or a script 
starting with  a  line  of  the form

以上exec系列函数区别:

1、带 l 的exec函数:execl、execlp、execle,表示后边的参数以可变参数的形式给出且都以一个空指针结束。

示例:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void)
{
    printf("entering main process---\n");
    execl("/bin/ls","ls","-l",NULL);
    printf("exiting main process ----\n");
    return 0;
}

结果:

entering main process---
总用量 24
-rwxr-xr-x 1 xcl xcl 16456 6月   7 09:11 execl
-rw-r--r-- 1 xcl xcl   215 6月   7 09:11 execl.c

利用 execl 将当前进程 main 替换掉,所有最后那条打印语句不会输出。 

2、带 p 的 exec 函数:execlp、execvp,表示第一个参数 path 不用输入完整路径,只有给出命令名即可,它会在环境变量 PATH 当中查找命令

示例:

当不带 p 但没给出完整路径时:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void)
{
    printf("entering main process---\n");
    execl("ls","-l",NULL);
    printf("exiting main process ----\n");
    return 0;
}

结果:

entering main process---
exiting main process ----

结果显示找不到,所有替换不成功,main进程继续执行

现在带 p:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void)
{
    printf("entering main process---\n");
    execlp("ls","-l",NULL);
    printf("exiting main process ----\n");
    return 0;
}

 结果:

entering main process---
main  main.c

替换成功。

3、不带 l 的 exec 函数:execv、execvp 表示命令所需的参数以 char *arg[] 形式给出且 arg 最后一个元素必须是 NULL 。

示例:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void)
{
    printf("entering main process---\n");
    int ret;
    char *argv[] = {"ls","-l",NULL};
    ret = execvp("ls",argv);
    if(ret == -1)
        perror("execl error");
    printf("exiting main process ----\n");
    return 0;
}

结果:

entering main process---
总用量 24
-rwxr-xr-x 1 xcl xcl 16512 6月   7 09:21 main
-rw-r--r-- 1 xcl xcl   306 6月   7 09:21 main.c

进程替换成功。

4、带 e 的 exec 函数:execle 表示,将环境变量传递给需要替换的进程

从上述的函数原型中我们发现:

extern char **environ;

此处的 environ 是一个指针数组,它当中的每一个指针指向的 char 为 “XXX=XXX”

environ 保存环境信息的数据可以 env 命令查看:

QQ截图20130712231740

它由 shell 进程传递给当前进程,再由当前进程传递给替换的新进程。

示例:execle.c

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char *argv[])
{
    //char * const envp[] = {"AA=11", "BB=22", NULL};
    printf("Entering main ...\n");
    int ret;
    ret =execl("./hello", "hello", NULL);
    //execle("./hello", "hello", NULL, envp);
    if(ret == -1)
        perror("execl error");
    printf("Exiting main ...\n");
    return 0;
}

 hello.c

#include <unistd.h>
#include <stdio.h>
extern char** environ;

int main(void)
{
    printf("hello pid=%d\n", getpid());
    int i;
    for (i=0; environ[i]!=NULL; ++i)
    {
        printf("%s\n", environ[i]);
    }
    return 0;
}

结果:

QQ截图20130712232918

可知原进程确实将环境变量信息传递给了新进程。

那么现在我们可以利用execle函数自己给的需要传递的环境变量信息:

示例程序:execle.c

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char *argv[])
{
    char * const envp[] = {"AA=11", "BB=22", NULL};
    printf("Entering main ...\n");
    int ret;
    //ret =execl("./hello", "hello", NULL);
    ret =execle("./hello", "hello", NULL, envp);
    if(ret == -1)
        perror("execl error");
    printf("Exiting main ...\n");
    return 0;
}

hello.c

#include <unistd.h>
#include <stdio.h>
extern char** environ;

int main(void)
{
    printf("hello pid=%d\n", getpid());
    int i;
    for (i=0; environ[i]!=NULL; ++i)
    {
        printf("%s\n", environ[i]);
    }
    return 0;
}

结果:

Entering main ...
hello pid=13886
AA=11
BB=22 

 确实将给定的环境变量传递过来了。

转载:exec系列函数(execl,execlp,execle,execv,execvp)使用_ning1350的博客-CSDN博客_execlp和execvp

(SAW:Game Over!)

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值