Chinaunix首页 | 论坛 | 博客
  • 博客访问: 418730
  • 博文数量: 121
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 1393
  • 用 户 组: 普通用户
  • 注册时间: 2014-03-11 12:17
个人简介

www.vibexie.com vibexie@qq.com

文章分类

全部博文(121)

文章存档

2015年(55)

2014年(66)

我的朋友

分类: C/C++

2014-09-20 00:43:39

fork()函数通过系统调用创建一个与原来进程(父进程)几乎完全相同的进程(子进程是父进程的副本,它将获得父进程数据空间、堆、栈等资源的副本。注意,子进程持有的是上述存储空间的“副本”,这意味着父子进程间不共享这些存储空间。linux将复制父进程的地址空间内容给子进程,因此,子进程有了独立的地址空间。),也就是这两个进程做完全相同的事。

在fork后的子进程中使用exec函数族,可以装入和运行其它程序(子进程替换原有进程,和父进程做不同的事)。

fork创建一个新的进程就产生了一个新的PID,exec启动一个新程序,替换原有的进程,因此这个新的被 exec 执行的进程的PID不会改变(和调用exec的进程的PID一样)。

exec函数族:


点击(此处)折叠或打开

  1. #include <unistd.h>
  2. extern char **environ;
  3. int execl(const char *path,
  4.           const char *arg, ...);
  5. int execlp(const char *file,
  6.            const char *arg, ...);
  7. int execle(const char *path,
  8.            const char *arg,
  9.            ...,
  10.            char * const envp[]);
  11. int execv(const char *path,
  12.           char *const argv[]);
  13. int execvp(const char *file,
  14.            char *const argv[]);
  15. int execve(const char *file,
  16.             char *const argv[],
  17.             char *const envp[]);
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. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <unistd.h>
  7. #include <string.h>
  8. #include <errno.h>

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

运行结果(linux):

点击(此处)折叠或打开

  1. 1------------execl------------
  2. . .. .deps exec exec.o .libs Makefile
  3. 2------------execv------------
  4. . .. .deps exec exec.o .libs Makefile
  5. 3------------execlp------------
  6. . .. .deps exec exec.o .libs Makefile
  7. 4------------execvp------------
  8. . .. .deps exec exec.o .libs Makefile
  9. 5------------execle------------
  10. . .. .deps .libs Makefile exec exec.o
  11. 6------------execve-----------
  12. . .. .deps .libs Makefile exec exec.o
  13. 按回车继续!


阅读(1272) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~