系统调用execve()对当前进程进行替换,替换者为一个指定的程序,其参数包括文件名(filename)、参数列表(argv)以及环境变量(envp)。
exec函数族不止一个,但它们大致相同,在 Linux中,它们分别是:execl,execlp,execle,execv,execve和execvp
一个进程一旦调用exec类函数,它本身就"死亡"了,系统把代码段替换成新的程序的代码,废弃原有的数据段和堆栈段,并为新程序分配新的数据段与堆栈段,唯一留下的,就是进程号,也就是说,对系统而言,还是同一个进程,不过已经是另一个程序了。(不过exec类函数中有的还允许继承环境变量之类的信息。)
在Linux中使用exec函数族主要有两种情况:
● 当进程认为自己不能再为系统和用户做出任何贡献时,就可以调用exec函数族中的任意一个函数让自己重生。
● 如果一个进程想执行另一个程序,那么它就可以调用fork()函数新建一个进程,然后调用exec函数族中的任意一个函数,这样看起来就像通过执行应用程序而产生了一个新进程(这种情况非常普遍)。
execl() 执行 ls -l
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int main(int argc ,char* argv[]) {
pid_t pid;
if ((pid=fork())==-1)
printf("error");
else if (pid==0)
execl("/bin/ls","ls","-l",argv[1],(char *)0);
else
printf("father ok\n");
}
利用函数execle,将环境变量添加到新建的子进程中去
#include<unistd.h>
#include<stdio.h>
int main()
{
char *envp[]={"PATH=/tmp","USER=sun",NULL};
if(fork()==0){
if(execle("/usr/bin/env","env",NULL,envp)<0){
printf("error");
}
}
}
运行结果:
用execve()函数执行自己写的新程序
感谢这篇博客 https://blog.csdn.net/str999_cn/article/details/78693438
//文件my.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]);
}
//文件exec.c,使用execve()函数调用my
#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("my",argv, environ);
//以下代码永远得不到运行的机会
printf("process never go to here!\n");
exit(0);
default:
printf("Parent process is running\n");
break;
}
wait(&stat_val);
exit(0);
}
运行结果截图:
无参数时:
有参数时: