使用fork或vfork创建子进程后,子进程通常会调用exec函数来执行另外一个程序。系统调用exec用于执行一个可执行程序以代替当前进程的执行映像。
exec调用没有生成新进程。一个进程一旦调用exec函数,它本身就“死亡”了,系统把代码段替换成新的程序的代码,废弃原有的数据段和堆栈段,并为新程序分配新的数据段和堆栈段,唯一保留的就是进程ID。也就是说,对系统而言,还是同一个进程,不过执行的已经是另外一个程序了。
//文件processimage.c,用以替换进程映像的程序
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main(int argc,char *argv[],char **environ)
{
int i;
printf("I am a process image!\n");
printf("ProcessImage:pid = %d, parentpid = %d\n", getpid(), getppid());
printf("ProcessImage:uid = %d,gid = %d\n", getuid(), getgid());
for(i=0; i< argc; i++)
printf("argv[%d]:%s\n",i ,argv[i]);
}
//文件execve.c,使用execve()函数调用processimage
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main(int argc,char * argv[],char ** environ)
{
pid_t pid;
int stat_val;
printf("Exec example!\n");
pid = fork();
switch(pid) {
case -1:
perror("Process Creation failed\n");
exit(1);
case 0:
printf("Child process is running\n");
printf("Execve pid = %d ,parentpid = %d\n",getpid(),getppid());
printf("Execve uid = %d,gid =%d\n",getuid(),getgid());
execve("processimage",argv, environ);
//以下代码永远得不到运行的机会
printf("process never go to here!\n");
exit(0);
default:
printf("Parent process is running\n");
break;
}
wait(&stat_val);
exit(0);
}
先编译第一个程序,生成可执行文件:gcc -o processimage processimage.c。再编译第二个程序:gcc -o execve execve.c。运行结果为:
从以上结果看,执行新程序的进程保持了原来进程的进程ID、父进程ID、实际用户ID和实际组ID。同时,调用新的可执行程序后,原有的子进程的映像被替代,不再被执行。子进程永远不会执行到printf("process never go to here.\n");,因为子进程已经被新的执行映像所替代。