Linux进程观:简单性如何成就强大性(六)

1. 进程程序替换

fork() 之后,⽗⼦各⾃执⾏⽗进程代码的⼀部分如果⼦进程就想执⾏⼀个全新的程序呢?进程的程序
替换来完成这个功能! 程序替换是通过特定的接⼝,加载磁盘上的⼀个全新的程序(代码和数据),加载到调⽤进程的地址空间 中!

1.1. 替换原理

⽤fork创建⼦进程后执⾏的是和⽗进程相同的程序(但有可能执⾏不同的代码分⽀),⼦进程往往要调⽤⼀种exec函数以执⾏另⼀个程序。当进程调⽤⼀种exec函数时,该进程的⽤⼾空间代码和数据完全被新程序替换,从新程序的启动例程开始执⾏。调⽤exec并不创建新进程,所以调⽤exec前后该进程的id并未改变。

1.2. 替换函数

其实有六种以exec开头的函数,统称exec函数:

cpp 复制代码
#include <unistd.h>

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[]);

1.2.1. 函数解释

  • 这些函数如果调⽤成功则加载新的程序从启动代码开始执⾏,不再返回。
  • 如果调⽤出错则返回-1。
  • 所以exec函数只有出错的返回值⽽没有成功的返回值。

1.2.2. 命名解释

这些函数原型看起来很容易混,但只要掌握了规律就很好记。

  • l(list):表示参数采用列表
  • v(vector):参数采用数组
  • p(PATH):自动搜索环境变量PATH
  • e(env):表示自己维护环境变量

|--------|------|-------|------------|
| 函数名 | 参数格式 | 是否带路径 | 是否使用当前环境变量 |
| execl | 列表 | 否 | 是 |
| execlp | 列表 | 是 | 是 |
| execle | 列表 | 否 | 自己传环境变量 |
| execv | 数组 | 否 | 是 |
| execvp | 数组 | 是 | 是 |
| execve | 数组 | 否 | 自己传环境变量 |

1.2.3. exec系列函数的举例

一. execl
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execl("/usr/bin/ls", "ls","-a","-l",NULL);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

二. execlp
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execlp("ls", "ls","-a","-l",NULL);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

三. execle
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	char *const my_envp[]={"MYPATH=11225566"};
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execle("/usr/bin/ls", "ls","-a","-l",NULL, my_envp);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

四. execv
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	char *const my_envp[]={"MYPATH=11225566"};
	char *const my_argv[]={"ls","-a","-l",NULL};
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execv("/usr/bin/ls", my_argv);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

五. execvp
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	char *const my_envp[]={"MYPATH=11225566"};
	char *const my_argv[]={"ls","-a","-l",NULL};
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execvp("ls", my_argv);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

六. execve
cpp 复制代码
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>

int main()
{
	char *const my_envp[]={"MYPATH=11225566"};
	char *const my_argv[]={"ls","-a","-l",NULL};
	int id = fork();
	if(id == 0)
	{
		printf("我是一个子进程,我将被替换.");
		execve("/usr/bin/ls", my_argv, my_envp);
	}else{
		printf("我是父进程,我不进行进程替换\n");
	}
	return 0;
}

运行结果:

事实上,只有execve是真正的系统调⽤,其它五个函数最终都调⽤execve,所以execve在man⼿册 第2节, 其它函数在man⼿册第3节。这些函数之间的关系如下图所⽰。
下图exec函数簇 ⼀个完整的例⼦:

相关推荐
Johny_Zhao2 天前
OpenClaw安装部署教程
linux·人工智能·ai·云计算·系统运维·openclaw
YuMiao2 天前
gstatic连接问题导致Google Gemini / Studio页面乱码或图标缺失问题
服务器·网络协议
chlk1233 天前
Linux文件权限完全图解:读懂 ls -l 和 chmod 755 背后的秘密
linux·操作系统
舒一笑3 天前
Ubuntu系统安装CodeX出现问题
linux·后端
改一下配置文件3 天前
Ubuntu24.04安装NVIDIA驱动完整指南(含Secure Boot解决方案)
linux
碳基沙盒3 天前
OpenClaw 多 Agent 配置实战指南
运维
深紫色的三北六号4 天前
Linux 服务器磁盘扩容与目录迁移:rsync + bind mount 实现服务无感迁移(无需修改配置)
linux·扩容·服务迁移
SudosuBash4 天前
[CS:APP 3e] 关于对 第 12 章 读/写者的一点思考和题解 (作业 12.19,12.20,12.21)
linux·并发·操作系统(os)
哈基咪怎么可能是AI4 天前
为什么我就想要「线性历史 + Signed Commits」GitHub 却把我当猴耍 🤬🎙️
linux·github
十日十行5 天前
Linux和window共享文件夹
linux