您的位置:首页 > 编程语言

在进程中运行新代码 execl、execle、execlp、execv、execve和execvp函数

2014-10-10 12:46 477 查看
摘要:本文主要讲述如何在进程中运行新代码,以及exec系列函数的基本使用方法.

在进程中运行新代码

用函数fork创建子进程后,如果希望在当前子进程中运行新的程序,可以调用exec函数执行另一个程序.当进程调用exec函数时,该进程用户空间资源(正文、数据、堆和栈)完全由新程序替代,新程序则从main函数开始执行.因为调用exec函数并没有创建新的进程,所以前后的进程ID并没有改变,也即内核信息基本不做修改.

exec系列函数共有7函数可供使用,这些函数的区别在于:指示新程序的位置是使用路径还是文件名,如果是使用文件名,则在系统的PATH环境变量所描述的路径中搜索该程序;在使用参数时使用参数列表的方式还是使用argv[]数组的方式.

1.exec系列函数

函数定义:

#include <unistd.h>

int execl(const char *pathname, const char *arg0, ... /* (char *)0 */ );

int execv(const char *pathname, char *const argv[]);

int execle(const char *pathname, const char *arg0, .../* (char *)0, char *const envp[] */ );

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

int execlp(const char *filename, const char *arg0, ... /* (char *)0 */ );

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

int fexecve(int fd, char *const argv[], char *const envp[]);

返回值:如果执行成功将不返回,否则返回-1,失败代码存储在errno中.

前4个函数取路径名作为参数,后两个是取文件名作为参数,最后一个是以一个文件描述符作为参数.

2.函数具体分析

当指定filename作为参数时:

1)如果filename中包含/,则将其视为路径名.

2)否则就按PATH环境变量,在它所指的各目录搜寻可执行文件.

2.1 execl()函数

int execl(const char *pathname, const char *arg0, ... /* (char *)0 */ );

execl()函数用来执行参数path字符串所指向的程序,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须是空指针以标志参数列表为空.

例子1:演示exec()函数的基本使用.

#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
#include <stdlib.h>
int main()
{
        pid_t pid;
        pid = fork();
        if(pid<0)
        {
                printf("error fork:%m\n");
                exit(-1);
        }
        else if(pid==0)
        {
                //
                execl("/bin/ls","ls","-l","/etc",(char *)0);
        }
        else
        {
                printf("parent process\n");
        }
        return 0;
}

输出:




2.2 execle()函数

int execle(const char *pathname, const char *arg0, .../* (char *)0, char *const envp[] */ );

execle()函数用来执行参数path字符串所指向的程序,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须指向一个新的环境变量数组,即新执行程序的环境变量.

例子2:

#include <unistd.h>
int main(int argc, char *argv[], char *env[])
{
        execle("/bin/ls","ls","-l", "/etc",(char *)0,env);
        return 0;
}

输出:




2.3 execlp()函数

int execlp(const char *filename, const char *arg0, ... /* (char *)0 */ );

execlp()函数会从PATH环境变量所指的目录中查找文件名为第一个参数指示的字符串,找到后执行该文件,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须是空指针.

例子3:

#include <unistd.h>

int main()
{
        execlp("ls", "ls", "-l", "/etc", (char *)0);
        return 0;
}

输出:




2.4 execv()函数

int execv(const char *path, char *const argv[]);

execv()函数函数用来执行参数path字符串所指向的程序,第二个为数组指针维护的程序参数列表,该数组的最后一个成员必须是空指针.

例子4:

#include <unistd.h>

int main()
{
        char *argv[] = {"ls", "-l", "/etc", (char *)0};
        execv("/bin/ls", argv);
        return 0;
}

输出:




2.5 execvp()函数

int execvp(const char *file, char *const argv[]);

execvp()函数会从PATH环境变量所指的目录中查找文件名为第一个参数指示的字符串,找到后执行该文件,第二个及以后的参数代表执行文件时传递的参数列表,最后一个成员必须是空指针.

例子5:

#include <unistd.h>

int main()
{
        char *argv[] = {"ls", "-l", "/etc", (char *)0};
        execvp("ls", argv);
        return 0;
}

输出:




几个函数之间非常相似,没有找到更好的方法记忆,通过简单的例子暂时了解一下它们之间的不同调用方式.字母联想也不靠谱:

字母p表示该函数取filename作为参数,并且用PATH环境变量寻找可执行文件.

字母l表示该函数取一个参数表,它与字母v互斥.

字母v表示该函数取一个argv[]矢量.

字母e表示该函数取envp[]数组.

待续......

笔者:个人能力有限,只是学习参考...读者若发现文中错误,敬请提出.

----------------------------------------------------------------------------------------------------------------------------------------------------------[b]-----------------------------------------勿在浮沙筑高台,静下心来,慢慢地沉淀---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------[/b]
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐