现在的位置: 首页 > 综合 > 正文

exec系列函数

2018年04月18日 ⁄ 综合 ⁄ 共 3509字 ⁄ 字号 评论关闭

用fork函数创建子进程后,如果希望在当前子进程中运行新程序,则可以调用exec系列函数。当进程调用exec系列函数中的任意一个时,该进程代码段、数据段内容完全由新程序替代。因为调用exec并不创建新进程,所以前后的进程号等相关信息并不发生变化。exec只是用新程序替换了当前进程的正文、数据、堆和栈段。


int   exec…装入和运行其它程序:
int   execl(  char *pathname,char *arg0,char *arg1,...,char *argn,NULL)
int   execle( char *pathname,char *arg0,char *arg1,...,char *argn,NULL,char *envp[])
int   execlp( char *pathname,char *arg0,char *arg1,...,NULL)
int   execlpe(char *pathname,char *arg0,char *arg1,...,NULL,char *envp[])
int   execv(  char *pathname,char *argv[])
int   execve( char *pathname,char *argv[],char *envp[])
int   execvp( char *pathname,char *argv[])
int   execvpe(char *pathname,char *argv[],char *envp[])

execl(执行文件)
表头文件  #include<unistd.h>
定义函数  int execl(const char * path,const char * arg,....);
函数说明  execl()用来执行参数path字符串所代表的文件路径,接下来的参数
          代表执行该文件时传递过去的argv[0]argv[1]……,最后一个参数必须用空指针(NULL)作结束。
返回值    如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno中。
范例
#include<unistd.h>
main()
{
execl(“/bin/ls”,”ls”,”-al”,”/etc/passwd”,(char * )0);
}
执行 
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd


execlp(从PATH 环境变量中查找文件并执行)
表头文件  #include<unistd.h>
定义函数  int execlp(const char * file,const char * arg,……);
函数说明  execlp()会从PATH 环境变量所指的目录中查找符合参数file的文件名,找到后执行该文件,然后将
          第二个以后的参数当做该文件的argv[0]argv[1]……,最后一个参数必须用空指针(NULL)作结束。
返回值    如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
范例
   
#include<unistd.h>
main()
{
execlp(“ls”,”ls”,”-al”,”/etc/passwd”,(char *)0);
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd


execv(执行文件)
表头文件  #include<unistd.h>
定义函数  int execv (const char * path, char * const argv[ ]);
函数说明  execv()用来执行参数path字符串所代表的文件路径,与execl()不同的地方在于
          execv()只需两个参数,第二个参数利用数组指针来传递给执行文件。
返回值    如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
范例
   
#include<unistd.h>
main()
{
char * argv[ ]={“ls”,”-al”,”/etc/passwd”,(char*) };
execv(“/bin/ls”,argv);
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd


execve(执行文件)
表头文件  #include<unistd.h>
定义函数  int execve(const char * filename,char * const argv[ ],char * const envp[ ]);
函数说明  execve()用来执行参数filename字符串所代表的文件路径,
          第二个参数系利用数组指针来传递给执行文件,argv要传递给程序的完整参数列表,
          包括argv0],它一般是执行程序的名字;最后一个参数则为传递给执行文件的新环境变量数组。
返回值    如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。

1.exec家族一共有六个函数,分别是:

(1)int execl(const char *path, const char *arg, ......);

(2)int execle(const char *path, const char *arg, ...... , char * const envp[]);

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

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

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

(6)int execlp(const char *file, const char *arg, ......);

其中只有execve是真正意义上的系统调用,其它都是在此基础上经过包装的库函数。

    exec函数族的作用是根据指定的文件名找到可执行文件,并用它来取代调用进程的内容,换句话说,就是在调用进程内部执行一个可执行文件。这里的可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。

与一般情况不同,exec函数族的函数执行成功后不会返回,因为调用进程的实体,包括代码段,数据段和堆栈等都已经被新的内容取代,只留下进程ID等一些表面上的信息仍保持原样,颇有些神似"三十六计"中的"金蝉脱壳"。看上去还是旧的躯壳,却已经注入了新的灵魂。只有调用失败了,它们才会返回一个-1,从原程序的调用点接着往下执行。

2.它们之间的区别:

第一个区别是:

前四个取路径名做为参数,后两个取文件名做为参数,如果文件名中不包含 “/” 则从PATH环境变量中搜寻可执行文件, 如果找到了一个可执行文件,但是该文件不是连接编辑程序产生的可执行代码文件,则当做shell脚本处理。

第二个区别:

前两个和最后一个函数中都包括“ l ”这个字母 ,而另三个都包括“ v ”, " l "代表 list即表 ,而" v "代表 vector即矢量,也是是前三个函数的参数都是以list的形式给出的,但最后要加一个空指针,如果用常数0来表示空指针,则必须将它强行转换成字符指针,否则有可能出错。,而后三个都是以矢量的形式给出,即数组。

最后一个区别:

与向新程序传递环境变量有关,如第二个和第四个以e结尾的函数,可以向函数传递一个指向环境字符串指针数组的指针。即自个定义各个环境变量,而其它四个则使用进程中的环境变量。

3.实例讲解:

(1)在平时的编程中,如果用到了exec函数族,一定记得要加错误判断语句。先判断execl的返回值,如果出错,可以用perror( )函数打印出错误信息。

如:if (execl(“path”,”..””(char *)0) < 0)

    {

       perror(“execl error!”);

   }

如果调用出错,可输出:execl error!: 错误原因   这样可方便查找出错原因

(2)注意下面书写格式:

先定义一个指针数组:char *argv[]={“ls”,”-l”,(char *)0}

用execv调用ls:    execv(“/bin/ls”,argv)

 如果用execvp

execvp(“ls”,argv)      //直接写ls就可以了

注意:

execl调用shell 时,要在shell脚本中指明使用的shell版本:#!/bin/bash。在命令行下执行shell脚本,系统为它自动打开一个shell,在程序中没有shell,在调用shell脚本时,会出错,所以要在shell脚本中先打开shell。

抱歉!评论已关闭.