Linux之exec函数族

exec函数族

  1. fork函数创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数以执行另一个函数。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并没改变
  2. 将当前的进程.text、.data替换为所要加载的程序的.text、.data,然后让进程从新的.text第一条指令开始执行,但进程ID不变,换核不换壳
    其中有六种以exec开头的函数,统称exec函数
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 execve(const char *path, char *const argv[], char *const envp[]);

execlp 函数 (p表示在函数运行期间,要用到PATH环境变量)

【加载一个进程,借助PATH环境变量】

int execlp(const char *file, const char *arg,...);

【返回值】成功:无返回;失败:-1
【参数一】要加载的程序的名字。该函数需要配合PATH环境变量来使用,当PATH中所有目录搜索后没有参数一则出错返回
该函数通常用来调用系统程序。如:ls、date、cp、cat等命令

#include <stdio.h>
#include <unistd.h>
int main()
{
	pid_t pid;
	pid = fork();
	if(pid < 0)
	{
		return -1;
	}
	else if(pid > 0)
	{
		sleep(1);
		printf("I am parent\n");
	}
	else
	{
		//相当于在终端执行 ls /home/wlr -l
		execlp("ls", "ls","/home/wlr", "-l", NULL);
	}
	return 0;
}

在这里插入图片描述

execl 函数 (l表示list,也就是参数列表)

【加载一个进程,通过 路径+程序名 来加载】

int execl(const char *path, const char *arg,...);

【返回值】成功:无返回;失败:-1

#include <stdio.h>
#include <unistd.h>
int main()
{
	pid_t pid;
	pid = fork();
	if(pid < 0)
	{
		return -1;
	}
	else if(pid > 0)
	{
		sleep(1);
		printf("I am parent\n");
	}
	else
	{
		//相当于在终端执行 ls /home/wlr -l
		execl("/bin/ls", "ls","/home/wlr", "-l", NULL);
	}
	return 0;
}

【比较】

execlp("ls", "ls", "/", "-l", NULL);  //使用程序名在PATH中搜索
execl("/bin/ls", "ls", "/", "-l", NULL);   //使用参数一给的绝对路径搜索
execl("./while", "ls","/home/wlr", "-l", NULL);  //执行自定义程序

【练习】将当前目录的ls信息,打印到文件中
这里会用到dup2函数,简单介绍一下dup2函数
在这里插入图片描述

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
	int fd;
	fd = open("ps.out", O_WRONLY | O_CREAT, 0644);
	if(fd < 0)
	{
		perror("open error\n");
		return -1;
	}
	dup2(fd, STDOUT_FILENO);
	execlp("ls", "ls", "-l", NULL); //dup2(3, 1) fd, stdout
	perror("exec error\n"); //exec函数只有调用失败才会返回,调用成功便不会执行exec后面的代码
	exit(0);
	close(fd);
	
	return 0;
}

在这里插入图片描述

exec函数族一般规律

exec函数一旦调用成功即执行新的程序,不返回。只有失败才返回,错误值为-1.所以我们通常直接在exec函数调用后直接调用perror()和exit(),无须if判断
【exec函数族后缀】

- l(list):命令行参数
- p(path):搜索file时使用path变量
- v(vector):使用命令行参数数组
- e(environment):使用环境变量数组,不使用进程原有的环境变量,设置新加载程序运行的环境变量

事实上,只有execve是真正的系统调用,其他五个函数最终都调用execve。这些函数之间关系如下
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux中,exec函数是用于执行一个新程序的函数。这个函数包括以下几个函数: 1. execl():执行一个指定的程序,它使用一个可变参数列表来传递程序名和参数列表。这个函数的原型如下: ```c int execl(const char *path, const char *arg, ...); ``` 2. execlp():与execl()函数相似,但它会在$PATH环境变量指定的路径中查找要执行的程序。这个函数的原型如下: ```c int execlp(const char *file, const char *arg, ...); ``` 3. execle():与execl()函数相似,但它还可以指定程序执行时的环境变量。这个函数的原型如下: ```c int execle(const char *path, const char *arg, ..., char *const envp[]); ``` 4. execv():与execl()函数相似,但它使用一个字符指针数组来传递参数列表。这个函数的原型如下: ```c int execv(const char *path, char *const argv[]); ``` 5. execvp():与execv()函数相似,但它会在$PATH环境变量指定的路径中查找要执行的程序。这个函数的原型如下: ```c int execvp(const char *file, char *const argv[]); ``` 这些函数中,每个函数的第一个参数都是要执行的程序的路径或文件名。如果路径是一个相对路径,那么它是相对于当前工作目录的。其他参数则是程序的命令行参数。 这些exec函数的使用方法都差不多,一般来说,它们会替换当前进程,也就是说,调用它们后,原来的程序就被新的程序所替代,而且新的程序从main()函数开始执行,不会返回到原来的程序。因此,在使用这些函数时,一定要注意保存当前进程的状态。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值