exec族函数的作用
我们在fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不会创建新进程,因此前后进程的ID并没有改变。在调用进程内部执行一个可执行文件,既可以是二进制文件也可以是Linux下的任何可执行的脚本文件。
函数族
execl,execlp,execle,execv,execvp,execvpe
函数原型
返回值
exec函数族的函数执行成功后不会返回,调用失败后,会设置errno并返回-1,然后从原程序的调用点接着往下执行。
参数说明
path:可执行文件的路径名
arg:可执行程序所带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束。
file:如果参数file中包含/,将其视为路径名,否则就按PATH环境变量,在它所指定的目录寻找可执行文件。
以execl函数举例来说明
创建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.c 文件。
#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;
}
执行结果
通过execl函数在execl.c中去执行echoarg.c中的代码,打印出传过去的两个参数。
execlp函数
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
printf("before execlp\n");
if(execlp("ps","ps","-l",NULL) == -1){
printf("execl failed!\n");
perror("why");
}
printf("after execlp\n");
return 0;
}
运行结果
execlp函数,能通过环境变量PATH查找到可执行文件ps,不用加入文件的绝对路径。
system函数
#include <string.h>
#include <stdlib.h>
#include <sys/wait.h>
int main()
{
pid_t pid;
int data;
while(1){
printf("please input data\n");
scanf("%d",&data);
if(data==1){
pid=fork();
if(pid>0){
wait(NULL);
}
if(pid==0){
system("./changeData test.txt");
}
else{
printf("do nothing\n");
}
}
return 0;
}
popen函数
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(int argc,char**argv)
{
char ret[1024]={0};
FILE* fp;
fp=popen("ps","r");
int nread=fread(ret,1,1024,fp);
printf("read %d byte ret=%s\n",nread,ret);
return 0;
}
运行结果:popen函数可以获取运行输出的结果
PS:修改环境变量PATH方法:
通过修改环境变量,能够在其他路径下执行指定路径下的可执行文件,不需要"./"。