exec系列函数

用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 要传递给程序的完整参数列表,
          包括 argv 0 ],它一般是执行程序的名字;最后一个参数则为传递给执行文件的新环境变量数组。
返回值      如果执行成功则函数不会返回,执行失败则直接返回 -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。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值