exec函数族

本文详细解析了Linux系统中exec函数族的功能、实现方式及其与fork函数的关系,阐述了它们在创建新进程和执行新程序过程中的作用。重点介绍了exec函数族的六种类型、参数差异、区别以及如何通过参数调整执行行为,同时揭示了写时复制技术优化fork和exec函数结合使用的方法。
摘要由CSDN通过智能技术生成

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,从原程序的调用点接着往下执行。

现在我们应该明白Linux是下如何执行新程序的,每当有程序认为自己不能为系统和拥护做出任何贡献了,就发挥一点余热,调用任何一个exec,让自己以新的面貌重生;或者,更普遍的情况是,如果一个进程想执行另一个程序,它就可以fork出一个新进程,然后调用任何一个exec,这样看起来就好像通过执行应用程序而产生了一个新的进程一样。
事实上第二种情况被应用的如此普遍,以至于Linux专门为其做了优化,我们已经知道,fork会调用进程的所有内容原封不动的复制到新产生的子进程中去,这些复制的动作很消耗时间,而如果fork完之后我们马上就调用exec,这些辛辛苦苦复制来的东西又会立刻抹掉,这看起来非常的不划算于是人们设计了一种“写时复制”技术,使得fork结束后并不立刻复制父进程的内容,而是到了真正用的时候才复制,这样如果下一条是exec,它就不会白白的做无用功了,也就提高了效率。

2.它们之间的区别:
第一个区别是:
前四个取路径名做为参数,后两个取文件名做为参数,如果文件名中不包含 “/” 则从PATH环境变量中搜寻可执行文件, 如果找到了一个可执行文件,但是该文件不是连接编辑程序产生的可执行代码文件,则当做shell脚本处理。
第二个区别:
前两个和最后一个函数中都包括“ l ”这个字母 ,而另三个都包括“ v ”, ” l “代表 list即表 ,而” v “代表 vector即矢量,也是是前三个函数的参数都是以list的形式给出的,但最后要加一个空指针,如果用常数0来表示空指针,则必须将它强行转换成字符指针,否则有可能出错。,而后三个都是以矢量的形式给出,即数组。
最后一个区别:
与向新程序传递环境变量有关,如第二个和第四个以e结尾的函数,可以向函数传递一个指向环境字符串指针数组的指针。即自个定义各个环境变量,而其它四个则使用进程中的环境变量。

exec函数族装入并运行程序path/file,并将参数arg0(arg1, arg2, argv[], envp[])传递给子程序,出错返回-1.
在exec函数族中,后缀l、v、p、e指定函数将具有某种操作能力:


后缀 操作能力
l 希望接收以逗号分隔的参数列表,列表以NULL指针作为结束标志
v 希望接收到一个以NULL结尾的字符串数组的指针
p 是一个以NULL结尾的字符串数组指针,函数可以DOS的PATH变量查找子程序文件
e 函数传递指定参数envp,允许改变子进程的环境,无后缀e时,子进程使用当前程序的环境

具体例子:

1#ifdef HAVE_CONFIG_H
  2 #include <config.h>
  3#endif  4  5 #include <stdio.h>
  6 #include <stdlib.h>
  7 #include <unistd.h>
  8 #include <string.h>
  9 #include <errno.h>
 10 11int main(int argc, char *argv[])
 12{
 13//以NULL结尾的字符串数组的指针,适合包含v的exec函数参数 14char *arg[] = {"ls", "-a", NULL};
 15 16/**
 17   * 创建子进程并调用函数execl
 18   * execl 中希望接收以逗号分隔的参数列表,并以NULL指针为结束标志
 19*/ 20if( fork() == 0 )
 21  {
 22// in clild  23     printf( "1------------execl------------\n" );
 24if( execl( "/bin/ls", "ls","-a", NULL ) == -1 )
 25    {
 26       perror( "execl error " );
 27       exit(1);
 28    }
 29  }
 30 31/**
 32   *创建子进程并调用函数execv
 33   *execv中希望接收一个以NULL结尾的字符串数组的指针
 34*/ 35if( fork() == 0 )
 36  {
 37// in child  38     printf("2------------execv------------\n");
 39if( execv( "/bin/ls",arg) < 0)
 40    {
 41       perror("execv error ");
 42       exit(1);
 43    }
 44  }
 45 46/**
 47   *创建子进程并调用 execlp
 48   *execlp中
 49   *l希望接收以逗号分隔的参数列表,列表以NULL指针作为结束标志
 50   *p是一个以NULL结尾的字符串数组指针,函数可以DOS的PATH变量查找子程序文件
 51*/ 52if( fork() == 0 )
 53  {
 54// in clhild  55     printf("3------------execlp------------\n");
 56if( execlp( "ls", "ls", "-a", NULL ) < 0 )
 57    {
 58       perror( "execlp error " );
 59       exit(1);
 60    }
 61  }
 62 63/**
 64   *创建子里程并调用execvp
 65   *v 望接收到一个以NULL结尾的字符串数组的指针
 66   *p 是一个以NULL结尾的字符串数组指针,函数可以DOS的PATH变量查找子程序文件
 67*/ 68if( fork() == 0 )
 69  {
 70     printf("4------------execvp------------\n");
 71if( execvp( "ls", arg ) < 0 )
 72    {
 73       perror( "execvp error " );
 74       exit( 1 );
 75    }
 76  }
 77 78/**
 79   *创建子进程并调用execle
 80   *l 希望接收以逗号分隔的参数列表,列表以NULL指针作为结束标志
 81   *e 函数传递指定参数envp,允许改变子进程的环境,无后缀e时,子进程使用当前程序的环境
 82*/ 83if( fork() == 0 )
 84  {
 85     printf("5------------execle------------\n");
 86if( execle("/bin/ls", "ls", "-a", NULL, NULL) == -1 )
 87    {
 88       perror("execle error ");
 89       exit(1);
 90    }
 91  }
 92 93/**
 94   *创建子进程并调用execve
 95   * v 希望接收到一个以NULL结尾的字符串数组的指针
 96   * e 函数传递指定参数envp,允许改变子进程的环境,无后缀e时,子进程使用当前程序的环境
 97*/ 98if( fork() == 0 )
 99  {
100     printf("6------------execve-----------\n");
101if( execve( "/bin/ls", arg, NULL ) == 0)
102    {
103       perror("execve error ");
104       exit(1);
105    }
106  }
107return EXIT_SUCCESS;
108 }
//运行结果
1------------execl------------
.  ..  .deps  exec  exec.o  .libs  Makefile
2------------execv------------
.  ..  .deps  exec  exec.o  .libs  Makefile
3------------execlp------------
.  ..  .deps  exec  exec.o  .libs  Makefile
4------------execvp------------
.  ..  .deps  exec  exec.o  .libs  Makefile
5------------execle------------
.  ..  .deps  .libs  Makefile  exec  exec.o
6------------execve-----------
.  ..  .deps  .libs  Makefile  exec  exec.o
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zxnsirius

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值