exec族函数的函数作用:
我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不会创建新进程,所以前后进程ID并没有改变。
exec族函数的功能
在调用进程内部执行一个可执行文件。可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。
家族成员:
execl、execlp、execle、execv、execvp、execvpe
它们对应的函数原型:
#include <unistd.h>
extern char **environ;
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg,..., char * const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execvpe(const char *file, char *const argv[],char *const envp[]);
返回值:
exec族函数在执行成功后不会返回,调用失败时,则会设置errno并返回-1,然后从原程序的调用点接着往下执行。
参数说明:
path: 可执行文件的路径名字
arg:可执行程序带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束
file: 如果参数file中包含/,则就将其视为路径名,否则就按PATH环境变量,在它所指定的各目录中搜寻可执行文件。
execl函数:
include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execl(const char *path, const char *arg, ...);
int main(void)
{
printf("before execl\n");
if(execl("./bin/echoarg","echoarg","abc",NULL) == -1)
{
printf("execl failed!\n");
perror("why"); // perror 会把错误平均到字符后面
}
printf("after execl\n");
return 0;
}

此时我们再写一个echoarg.c的文件
#include <stdio.h>
int main(int argc,char *argv[])
{
int i = 0;
for(i = 0; i < argc; i++)
{
printf("argv[%d]: %s\n",i,argv[i]);
}
return 0;
}

然后再通过execl函数去调用存在的文件
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execl(const char *path, const char *arg, ...);
int main(void)
{
printf("before execl\n");
if(execl("./echoarg","echoarg","abc",NULL) == -1)
{
printf("execl failed!\n");
perror("why");
}
printf("after execl\n");
return 0;
}

当然啦,也可以调用ls, 调用前先查一下ls的绝对路径

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execl(const char *path, const char *arg, ...);
int main(void)
{
printf("before execl\n");
if(execl("/bin/ls","ls",NULL,NULL) == -1)
{
printf("execl failed!\n");
perror("why");
}
printf("after execl\n");
return 0;
}
输出结果

等同于执行了ls
如果要执行ls -l 呢? 答案是给execl传参即可

执行结果

获取系统时间:date
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execl(const char *path, const char *arg, ...);
int main(void)
{
printf("this pro get system date\n");
if(execl("/bin/date","date",NULL,NULL) == -1)
{
printf("execl failed!\n");
perror("why");
}
printf("after execl\n");
return 0;
}

execlp函数:
由于execl的路径,每次都要whereis 去查找一下绝对路径,于是想着什么方法能省略这一步,答案是execlp可以
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execlp(const char *file, const char *arg, ...);
int main(void)
{
printf("this pro get system date\n");
if(execlp("ps","ps",NULL,NULL) == -1)
{
printf("execl failed!\n");
perror("why");
}
printf("after execl\n");
return 0;
}

上面的exaclp函数带p,所以能通过环境变量PATH查找到可执行文件ps
execvp函数:

相当于用一个指针字符串数组把参数装起来
execv函数
include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
printf("this pro get system date\n");
char *argv[] = {"ps","-l",NULL};
if(execv("ps",argv) == -1)
{
printf("execl failed!\n");
perror("why");
}
printf("after execl\n");
return 0;
}
结果

需要加上绝对路径


总结就是加 p 不用加绝对路径, 不加p需要加绝对路径(/bin/xxx)