学习笔记——exec族函数详解(execl, execlp, execle, execv, execvp, execvpe )

exec族函数的定义

定义

exec函数族提供了一个在进程中启动另一个程序执行的方法。它可以根据指定的文件名或目录名找到可执行文件,并用它来取代原调用进程的数据段、代码段和堆栈段,在执行完之后,原调用进程的内容除了进程号外,其他全部被新的进程替换了。另外,这里的可执行文件既可以是二进制文件,也可以是Linux下任何可执行的脚本文件。

作用

我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不创建新进程,所以前后进程的ID并没有改变。

exec函数族有这几个: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环境变量,在它所指定的各目录中搜寻可执行文件。

exec族函数实例

上面的定义已经解释得差不多了,接下来我们通过几个实例看看exec族函数究竟是怎么使用的。

execl函数


#include <stdio.h>
int main(int argc,char **argv){
	int i;

	for(i = 0;i < argc;i++){
		printf("argv[%d]: %s\n",i,argv[i]);
	}
	return 0;
}

//这个demo非常简单,打印我们输入的参数,我们把这个文件命名为execlnewpro(可执行文件)

======================分割线==================================


#include <stdio.h>
#include <unistd.h>
int main(int argc ,char **argv){
	
	printf("before execl!\n");	
	if(execl("./execlnewpro","execlnewpro","abc",NULL) == -1){
		printf("execl filed!\n");
		
		perror("becasue");
	}
	printf("Execute after failure!\n");
	return 0;
}

下面这个demo使用execl函数,传入path也就是execlnewpro的路径(这里也可以写绝对路径),argv的第一个参数是我们需要的可执行文件名字,abc是第二个参数,结尾必须以NULL结尾。如果execl执行后返回-1,代表没有这个文件,会执行printf(“execl filed!\n”);这句话。我们可以通过perror来打印错误信息,那么紧接着会执行printf(“Execute after failure!\n”);,因为filed会原程序的调用点接着往下执行。 如果成功就直接去调用execlnewpro了。

插一句: perror 头文件 #include <stdio.h>
原型 void perror(const char *s);

打印结果如下图所示:
在这里插入图片描述


再比如说:

#include <stdio.h>
#include <unistd.h>

int main(int argc ,char **argv){
	
	printf("before execl!\n");	
	if(execl("/bin/ls","ls","-l",NULL) == -1){
		printf("execl filed!\n");
		
		perror("becasue");
	}
	printf("Execute after failure!\n");
	return 0;
}

我们通过pwd命令查看ls的绝对路径,是在/bin/ls,调用ls并带着-l参数。通过execl会直接去调用ls -l命令,显示当前列表信息
在这里插入图片描述

execlp函数

那么execl函数后面加一个p是什么意思呢?p:使用文件名,并从PATH环境进行寻找可执行文件 ,,,带p的函数包括execlp、execvp、execvpe,如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件。举个例子,PATH=/bin:/usr/bin

接着上面的demo我们继续,比如我现在想要获取时间,需要用到date命令,

#include <stdio.h>
#include <unistd.h>
int main(int argc ,char **argv){
	
	printf("before execl!\n");	
	if(execlp("date","date",NULL,NULL) == -1){
		printf("execl filed!\n");
		
		perror("becasue");
	}
	printf("Execute after failure!\n");
	return 0;
}

execlp函数会通过我们给出的data在系统变量中寻找文件,找到调用,否则error-1

插一句:whereis 想要查看命令,作用是查看命令的绝对路径比如whereis ls,可以查看ls的绝对路径
echo $ PATH Linux查看当前系统环境变来量
export PATH=$ PATH:(pwd查看当前的路径)
比如:
pwd(回车) /home/xxx/process
export PATH=$ PATH: /home/xxx/process ,这样就把当前路路径加到环境变量中了

结果上图:
在这里插入图片描述

execvp函数

那么带v的函数是什么意思呢? execv、execvp、execve,应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。

也就是在函数外面指定好指针数组,exec函数直接使用

看看demo


#include <stdio.h>
#include <unistd.h>

int main(void){
	
	char *argv[] = {"ls","-l",NULL};
	if(execvp("ls",argv) == -1){
		printf("execl filed!\n");
		
		perror("becasue");
	}
	printf("Execute after failure!\n");
	return 0;
}

//先指定好char *argv[] = {"ls","-l",NULL};,我们exec函数直接使用,一样的效果,目的一样,使用方法不一样

补充

那么我们使用fork函数和exec函数搭配使用是什么效果呢?之前我们说fork函数创建子进程也可以做不同的事情,那么这里就需要子进程返回后马上调用exec函数,不改变id去执行其他的可执行文件

学习笔记——Linux小应用修改配置文件
之前的章节写了修改配置文件的demo,那么在本节我们就把这个demo编译为demo7pro,编辑文件时config.txt

见代码:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>

int main(){
	pid_t pid;
	int count = 0;
	int data = 0;
	
	pid =getpid();
	while(1){	
		printf("Please input the data!\n");
		scanf("%d",&data);
		if(data == 1){	
			pid = fork();
			
			if(pid > 0){
				wait(NULL);
			}
			if(pid == 0){
				execl("./demo7pro","demo7pro","config.txt",NULL);	
			}		
		}else{
		printf("do nothing!\n");
		}	
	}
	return 0;
}

在这里我们使用execl函数直接去调用demo7pro,进而对config.txt的内容进行修改。而如果不使用execl函数,就要把之前章节的demo拿过来进行整合,代码量增多,且繁琐。

在这里插入图片描述

到这里config文件中的字段已经修改成功。

带e的一类exac函数,包括execle、execvpe,可以传递一个指向环境字符串指针数组的指针。 参数例如char *env_init[] = {“AA=aa”,”BB=bb”,NULL}; 带e表示该函数取envp[]数组,而不使用当前环境。

总结

l : 使用参数列表
p:使用文件名,并从PATH环境进行寻找可执行文件
v:应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。
e:多了envp[]数组,使用新的环境变量代替调用进程的环境变量

参考:https://blog.csdn.net/yychuyu/article/details/80173039?spm=1001.2101.3001.6650.3&depth_1-utm_relevant_index=4

  • 13
    点赞
  • 53
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

石子君

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值