当前位置: 首页 > news >正文

进程程序替换

fork() 之后,⽗⼦各⾃执⾏⽗进程代码的⼀部分如果⼦进程就想执⾏⼀个全新的程序呢?进程的程序

替换来完成这个功能!

程序替换是通过特定的接⼝,加载磁盘上的⼀个全新的程序(代码和数据),加载到调⽤进程的地址空间中!

 4.1 替换原理

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

 

 4.2 替换函数

其实有六种以exec开头的函数,统称exec函数:
man exec
#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[]);

4.2.1 函数解释

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

4.2.1 函数讲解

第一种:

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

知识点:对于接下来讲的exec序列函数中的知识点是通用的

1.一旦exec函数替换,就去执行新代码了,原始后面的代码已经不存在了, 

2.不做返回值判断,只要返回就是失败了。

引出问题:除了替换指令可以替换自己写的程序吗

 先随意写一个语言的代码 c++为例

怎么证明当前你在进行替换的时候,并没有创建新的进程呢,是同一个pid吗?

 

原理:没有在原替换过程,创建新程序,只是把当前进程的代码和数据用新的程序的代码与数据覆盖式的替换,此时子进程与父进程再也没有关系了,他们分离了。

第二种

execlp("ls","ls","-ln","-a",NULL);

第三种

#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main()
{printf("我的程序要运行了!\n");if(fork() == 0){printf("I am Child, My Pid Is: %d\n", getpid());sleep(1);char *const argv[] = {  //命令行参数表(char*const)"ls",(char*const)"-l",(char*const)"-a",NULL};execv("/usr/bin/ls", argv);exit(1);}waitpid(-1, NULL, 0);printf("我的程序运行完毕了\n");//return 0;
}

 小知识:所有进程所需要的命令行参数都是通过父进程用exec 传的命令行参数

第四种带有环境变量

相关问题

1. 关于 execve 更新环境变量后以前环境变量消失

execve 函数执行成功后,会用新程序完全替换当前进程的执行映像,包括代码段、数据段等。新程序启动时,使用的是通过 envp 参数传入的环境变量数组。如果传入的 envp 中不包含之前进程的某些环境变量,那么在新程序中这些之前的环境变量就不存在了。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main() {char *const argv[] = {"echo", "Hello"};char *const new_envp[] = {"MY_NEW_VAR=value",NULL};// 执行echo程序,传递新的环境变量数组,原进程环境变量未传入if (execve("/bin/echo", argv, new_envp) == -1) {perror("execve");exit(EXIT_FAILURE);}return 0;
}

在上述代码中,execve 执行 echo 程序时,只传入了自定义的环境变量数组 new_envp ,原进程的环境变量没有包含在内,所以 echo 程序运行时看不到原进程的环境变量,就好像之前的环境变量 “消失” 了。

2. 使用 putenv 更新环境变量示例

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main() {// 定义要设置的环境变量字符串char var_value[] = "MY_VAR=HelloWorld";// 使用putenv设置环境变量if (putenv(var_value) != 0) {perror("putenv");return EXIT_FAILURE;}// 获取并打印环境变量char *value = getenv("MY_VAR");if (value != NULL) {printf("MY_VAR value: %s\n", value);} else {printf("Failed to get MY_VAR\n");}return EXIT_SUCCESS;
}

在这个例子中,putenv 函数用于将 MY_VAR=HelloWorld 这个环境变量添加到当前进程的环境变量表中。putenv 函数接受一个形如 "变量名=变量值" 的字符串参数,如果设置成功,它不会返回错误值;若失败(比如内存分配问题等 ),则返回非零值。之后通过 getenv 函数获取并打印刚刚设置的环境变量的值。

3. 证明不更新环境变量子进程本身有

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main() {pid_t pid = fork();if (pid < 0) {perror("fork");return EXIT_FAILURE;} else if (pid == 0) {// 子进程char *value = getenv("PATH");if (value != NULL) {printf("子进程中PATH环境变量值: %s\n", value);} else {printf("子进程中获取PATH环境变量失败\n");}return EXIT_SUCCESS;} else {int status;pid_t wpid = waitpid(pid, &status, 0);if (wpid == -1) {perror("waitpid");return EXIT_FAILURE;}return EXIT_SUCCESS;}
}

在这段代码中,通过 fork 创建子进程。子进程中使用 getenv 函数获取 PATH 环境变量(没有对环境变量进行更新操作 )。由于子进程会继承父进程的环境变量,所以在子进程中可以获取到父进程传递下来的 PATH 环境变量的值,从而证明子进程本身在不更新环境变量的情况下,是有从父进程继承来的环境变量的。

 小知识:其实替换函数有七个 ,上面举例了六个,剩下的哪一个是系统调用的,也就是说,为什么那六个就算不写环境变量,也会本身自带,正是因为那六个是语言封装的,自动会调用系统自带的。

http://www.xdnf.cn/news/30097.html

相关文章:

  • 【橘子大模型】初探rag知识库的构建
  • Linux基础IO(八)之硬链接
  • 完整游戏排行榜系统实现
  • Redux Promise 中间件
  • C++ 数组 array ™实现动画效果全解析⚡YQW · Studio ⚡
  • Http基础
  • QML中的3D功能--自定义着色器开发
  • 硬件操作指南——ATK-MD0430 V20
  • 什么是超类实体和派生属性
  • JavaScript 变量语法扩展
  • C 语言联合与枚举:自定义类型的核心解析
  • Scade 语言词法介绍
  • 游戏引擎学习第235天:在 Windows 上初始化 OpenGL
  • 4N60-ASEMI开关电源与适配器专用4N60
  • 6.7 ChatGPT自动生成定时任务脚本:Python与Cron双方案实战指南
  • android测试依赖
  • Python番外——常用的包功能讲解和分类组合
  • GD32H7单片机使用segger_rtt,rtt-viewer看不到输出的问题,怎样解决?
  • 使用docker在manjaro linux系统上运行windows和ubuntu
  • 在统信UOS1060上新增备份到U盘
  • 【java实现+4种变体完整例子】排序算法中【基数排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格
  • Python----深度学习(全连接与链式求导法则)
  • Java中常见的锁synchronized、ReentrantLock、ReentrantReadWriteLock、StampedLock
  • MainActivity与RecActivity之间的双向数据传递详解
  • 从 0~1 保姆级 详细版 PostgreSQL 数据库安装教程
  • 数据库备份-docker配置主从数据库
  • k8s安装kubeadm
  • 探索大语言模型(LLM):Transformer 与 BERT从原理到实践
  • 回溯算法(2):全排列问题
  • 基于DeepSeek与Excel的动态图表构建:技术融合与实践应用