Linux之进程替换

news2025/1/11 13:02:51

创建子进程的目的

创建子进程的第一个目的是让子进程执行父进程对应的磁盘代码中的一部分, 第二个目的是让子进程想办法加载磁盘上指定的程序,让子进程执行新的代码和程序 

一是让子进程执行父进程代码的一部分, 比如:

  1 #include<stdio.h>  
  2 #include<unistd.h>  
  3 #include<sys/types.h>  
  4 #include<sys/wait.h>  
  5 int main()  
  6 {                                                                                                                                   
  7     pid_t id=fork();                  
  8     if(id==0)                          
  9     {                                  
 10         //执行子进程的内容             
 11     }                                  
 12     else                               
 13     {                                  
 14         //执行父进程的内容                                                   
 15         wait(NULL);                                                          
 16     }                                                                        
 17     return 0;                          
 18 }  

另外一个就是让子进程执行一个全新的程序, 比如说我们在其他路径下写的一些可执行程序.

第二种方法可称之为进程替换, 进程替换是父进程创建子进程之后让子进程执行其他程序的代码而不是执行父进程代码的一部分.

注意和进程切换不同, 进程切换是不同的进程都有一个属于自己的时间片, cpu每次只能执行一个进程的数据和代码, 所以为了保证多个进程能够正常的运行, cpu每次执行一个进程都只会执行时间片长度的时间, 时间到了就硬件上下文把信息保存在PCB, 然后换下一个进程到cpu里执行这叫进程切换.

如何实现进程替换 

如何来实现进程替换呢?

实现进程替换得用到函数execl:

平时执行程序的时候需要告诉bash可执行程序在哪里, 以及采用什么样的方式调用执行这个命令, 这里也是同样的道理, 在子进程中使用execl函数执行其他可执行程序时也需要给函数传相应的参数如可执行程序位置, 以及以什么样的方式来执行这个程序(cmd 选项1 选项2…)

参数path就是可执行程序所在的位置, 后面的可变参数列表是可执行程序所用的方法, 后面的...是可变参数列表, 该函数的参数个数由传参的个数决定(命令行里怎么写的,就怎么传参),最后必须以NULL结尾,表示参数传递完毕..


单进程版的程序替换代码:

系统指令本质上是可执行程序所以我们也可以让子进程执行系统中的指令: 

  1 #include<stdio.h>
  2 #include <unistd.h>
  3 int main()
  4 {
  5     printf("pid: %d, exec command begin\n",getpid());
  6     execl("/usr/bin/ls","ls","-l","-a",NULL);                                                      
  7     printf("pid: %d, exec command end\n",getpid());
  8     return 0;
  9 }

平时使用ls指令时可以添加一些选项比如说:-a -l等, 那调用函数时就得把指令和选项分开当成一个个字符串传递给这个函数并且参数的最后必须以NULL进行结尾:

 命令行中ls是ls --color=auto的别名, 所以会显示颜色: 

 

在代码中加上"--color=auto":

  

也可以执行自己写的可执行程序:

 myprocess.cc:

  1 #include <iostream>
  2 
  3 int main()
  4 {
  5     std::cout << "hello C++" << std::endl;
  6     std::cout << "hello C++" << std::endl;
  7     std::cout << "hello C++" << std::endl;
  8     std::cout << "hello C++" << std::endl;                                                         
  9     return 0;
 10 }

myprocess.cc生成的可执行文件命名为myprocess, 在 test.c中进行程序替换

test.c

  1 #include<stdio.h>
  2 #include <unistd.h>
  3 int main()
  4 {
  5     printf("pid: %d, exec command begin\n",getpid());
  6     //execl("/usr/bin/ls","ls","-l","-a","--color=auto",NULL);
  7     execl("./myprocess","./myprocess",NULL);                                                         
  8     printf("pid: %d, exec command end\n",getpid());
  9     return 0;
 10 }

myprocess这个程序没有选项所以这个程序执行的方法就是./myprocess, 这里的方法也可以直接写成process不需要添加上面的相对路径 

makefile: 

结果: 

这就说明execl函数既可以让程序执行系统自带的指令, 也可以让程序执行我们自己写的可执行程序

但是这个运行结果执行结果好像"不太对", 在execl函数后面我们还使用了一个printf函数, 但是程序执行完execl函数之后并没有执行printf函数, 那就要了解一下进程替换的原理.


多进程版程序替换代码

进程替换的原理

关于进程替换的返回值,execl只在发生错误的时候有返回值, 如果替换失败了就会返回-1: 

比如说下面的代码, 我们给execl函数传递一个不存在的路径, 就可以发现execl函数的返回值是-1 :

如果正常执行呢?

这里好像没有打印出来execl函数的返回值, 并且连printf函数都没有正常执行.

原因分析:

首先在没有执行fork函数之前操作系统中就只有一个父进程, 这个父进程有对应的数据区和代码区:

 

当执行fork函数之后操作系统中就会多出来一个子进程, 并且操作系统会以父进程的部分数据结构为这个子进程创建页表,虚拟地址空间和PCB, 所以子进程的页表也指向物理空间上的那块区域, 也就是说此时的子进程和父进程是共用内存上同一块数据区和代码区 

当子进程调用execl函数时会将磁盘上的程序B加载进内存并替换原来的子进程所指向的数据区和代码区, 如果此时子进程的这块数据区和代码区只有它自己使用, 则B程序的数据区的代码区会直接覆盖原来的, 但是进程之间是有独立性的, 当前内存中的程序A的数据区和代码区不仅仅子进程在使用父进程也再使用, 所以此时execl函数再发生替换时会发生写时拷贝, 操作系统会在内存上再开辟一个空间用来存放程序B的数据和代码再将子进程的页表指向新的空间, 这样子程序就能执行程序B的代码:

这个新空间和原来父进程的数据代码就没有任何关系了, 他是一个全新的内容所以执行完execl函数之后就不会再执行原来程序中execl函数后面的内容, 所以execl函数替换成功之后的返回值是什么也就不重要了.


有关进程替换的函数

实现程序替换不止execl函数还有execlp,execv,execvp,execle,execve函数

execl

调用接口:int execl(const char *path, const char argv, ...);

头文件:unistd.h

参数:path表示新程序在磁盘的地址, argv是一个可变参数列表, 需要将我们执行这个程序时向控制台输入的东西以空格为分割一段一段字符串传参, 最后以NULL结尾.

功能:通过程序位置和程序运行名加选项进行进程替换.

execlp

调用接口:int execlp(const char *file, const char *argv, ...);

头文件:unistd.h

参数:file表示新程序的文件名,argv与上面一样(注意传空指针),此时这个函数会在环境变量的位置中寻找相应可执行程序

功能:通过程序名和程序名加选项进行进程替换

 这个函数相较于execl多了一个p, 这个p表示的意思就是path也就是环境变量中的PATH.

 环境变量中的PATH记录着各种操作系统指令所在的地址, 所以使用这个函数替换程序时不需要传地址, 直接传程序名就行, 因为这个函数会自动在PATH所记录的路径查找这个程序, 这个函数的第一个参数就对应着程序名, 比如:

  1 #include <stdio.h>
  2 #include <unistd.h>
  5 int main()
  6 {
  7     printf("exec command begin:\n");
  8     execlp("ls","ls","-l","-a",NULL);                                                              
  9     return 0;
 10 }

第一个ls表示的是程序名, 第二个ls表示的是执行程序的方法, 也就是可变参数列表中的第一个参数


execv

调用接口:int execv(const char *path, char *const argv[]);

头文件:unistd.h

参数:path表示新程序在磁盘的地址. 这里的argv就不太一样了, 是一个指针数组, 每一个指针指向一个字符串, 最后的位置也必须是NULL, const修饰的是这个字符串数组, 数组内的元素不能改变.

功能:通过程序位置、程序名加选项组成的数组进行进程替换

 这个函数的v表示的是vector, 也就是说将程序的所有指令全部放入到数组中比如说下面的代码: 

    1 #include <stdio.h>
    2 #include <unistd.h>
    5 int main()
    6 {
    7     char *const arr[]={"ls","-a","-l",NULL};
    8     printf("exec command begin:\n");
    9     execv("/usr/bin/ls",arr);
   10     return 0;
   11 }

 


execvp

调用接口:int execvp(const char *file, char *const argv[]);  

头文件:unistd.h

参数:file表示新程序的文件名, argv与上面的execv一样。

功能:通过程序名和程序名加选项进行进程替换和环境变量表进行进程替换

    1  #include<stdio.h>
    2  #include<unistd.h>
    3  int main()
    4  {      
    5      char *const arr[]={"ls","-a","-l",NULL}; 
    6      printf("exec command begin:\n");                                                                                    
    7      execvp("ls",arr);
    8      return 0;
    9  }


几个函数之间非常相似, 可以通过字母辅助记忆:

字母p表示该函数取文件名file作为参数, 并且用环境变量寻找可执行文件, 不需要传递它在磁盘的位置
字母l表示该函数取一个list
字母v表示该函数取一个数组,可以理解为vector
字母e表示该函数取环境变量envp[]数组


 execle

这个函数的参数如下 

这个函数就多一个参数第三个参数表示的意思是环境变量,如果子进程要用到自定义环境变量或者系统的环境变量的话就可以用到第三个参数,比如说在当前路径下再创建一个文件: 

 先来看这样一段程序替换的代码:

//mytest.c 
 1 #include <stdio.h>
  2 #include <unistd.h>
  3 #include <sys/types.h>
  4 #include <sys/wait.h>
  5 #include<stdio.h>
  6 #include<unistd.h>
  7 
  8 
  9 int main()
 10 {
 11     pid_t id = fork();
 12     if(id == 0)
 13     {
 14         //child
 15         printf("pid :%d,exec command begin:\n",getpid());                                            
 16         int ret = execl("./myprocess","myprocess",NULL);
 17         if(ret == -1)
 18             printf("process replace fail\n");
 19         printf("pid : %d,exec command end,return value is %d.\n",getpid(),ret);
 20     }
 21     else
 22     {
 23         //father
 24         pid_t rid = waitpid(-1,NULL,0);
 25         if(rid > 0)
 26             printf("wait success, rid : %d\n",rid);
 27     }
 28     return 0;
 29 }
//myprocess.cc  
  1 #include <iostream>
    2 
W>  3 int main(int argc, char* argv[],char* env[])
    4 {
    5     for(int i = 0; env[i];i++)
    6     {
    7         std::cout << i << ":" << env[i] << std::endl;                                              
    8     }
    9     return 0;
   10 }

问题1: 执行后进程替换如期打印出了环境变量表, 那我们程序替换时, 子进程(myprocess)的环境变量从哪里来的? 是从父进程来的,它是父进程调用的, 也只能由父进程传递. 

父进程的环境变量从哪里来? 是从bash来的, mytest创建进程首先要成为bash的子进程, bash传递环境变量给它, 然后它才作为父进程去创建自己的子进程.

所以bash, mytest, myprocess 用的是一套环境变量:

可以看到孙子进程确实打印出来了自己在bash导入的环境变量.

现在用putenv在父进程中导入一个环境变量: 

 可以看到myprocess继承了mytest的环境变量, 但bash并没有更新MYVAL, 因为mytest中的MYVAL是在bash传入环境变量之后才更改的, bash不会有影响:

问题2: 环境变量被子进程继承下去是一种默认的行为, 不受程序替换的影响, 为什么?

父进程在创建子进程时, 子进程继承父进程的部分PCB,进程地址空间和页表, 进程地址空间上的虚拟地址, 通过页表映射到物理内存, 所以其实环境变量被子进程继承下去是一件很正常的事, 所以是通过地址空间让子进程继承父进程环境变量数据.

环境变量和命令行参数也是进程的数据, exec*程序不是会替换进程的代码和数据吗? 程序替换只替换新程序的代码和数据, 环境变量不会被替换.

所以其实不用所谓的execle之类的函数传递环境变量环境变量也是默认传递下去的.

问题3:子进程执行的时候获得的环境变量.

a. 将父进程的环境变量原封不动的传递给子进程.

1.直接用

environ变量也是可以直接打印出进程所对应的环境变量的, 不用env参数也能访问环境变量. 

2.直接传参

environ是char**类型的, execle的第三个参数是char* []类型, 是匹配的.

b.我们想传递自己的环境变量 , 我们可以直接构造自己的环境变量表, 给子进程传递,这个传递不是新增而是覆盖式地传递.

 

 c.新增传递

新增之前已经实现过, 想新增哪些环境变量在父进程中putenv添加即可.

结论: 程序替换可以将命令行参数和环境变量通过自己的参数, 传递给被替换的程序的main函数中.


execvpe

这个函数的使用就和之前完全类似了,e表示环境变量, p表示直接传环境变量名即可, v表示这里需要方法组成的数组, 那这个函数的参数就如下:


execve

上面讲了那么多的函数,但是这些函数都有一个共同的特点就是这些函数都是c语言函数提供的,系统提供了一个函数调用接口execve,这个函数的参数如下:

 上面介绍的六个函数最后都会被转化成execve:

 最终底层调的是一个接口, 那为什么还有这么多不一样的呢? 虽然接口形式是不一样的, 但是本质还是一样的, 主要还是为了满足各种调用的场景, 不一定自己在进程程序替换时, 程序名这个字符串就替代好了, 环境变量也不一定都是以表的形式呈现好了, (比如execl传的是一个个的选项,底层execve调用时就封装成了一个数组),所以直接选择合适的就可以.

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1240868.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ubuntu编译sqlite3并使用

SQLite3是一种轻量级的关系型数据库管理系统&#xff0c;它是在C语言基础上实现的。SQLite3具有许多优点&#xff0c;例如&#xff1a; 1.灵活&#xff1a;它可以在多种操作系统上运行&#xff0c;并且可以将多个数据库文件合并成一个文件。 2.易于使用&#xff1a;SQLite3使用…

循环队列详解!!c 语言版本(两种方法)双向链表和数组法!!

目录 1.什么是循环队列 2.循环队列的实现&#xff08;两种方法&#xff09; 第一种方法 数组法 1.源代码 2.源代码详解&#xff01;&#xff01; 1.创造队列空间和struct变量 2.队列判空 3.队列判满&#xff08;重点&#xff09; 4.队列的元素插入 5.队列的元素删除 …

2023亚太杯数学建模竞赛(亚太赛)选题建议+初步分析

如下为C君的2023亚太杯数学建模竞赛&#xff08;亚太赛&#xff09;选题建议初步分析&#xff1a; 提示&#xff1a;DS C君认为的难度&#xff1a;C<A<B&#xff0c;开放度&#xff1a;A<B<C。 以下为ABC题选题建议及初步分析&#xff1a; A题&#xff1a;Image…

读像火箭科学家一样思考笔记06_初学者之心

1. 专业化是目前流行的趋势 1.1. 通才&#xff08;generalist&#xff09;是指博而不精之人 1.2. 懂得的手艺越多&#xff0c;反而会家徒四壁 1.2.1. 希腊谚语 1.3. 这种态度代价很大&#xff0c;它阻断了不同学科思想的交融 2. 组合游戏 2.1. 某个行业的变革可能始于另一…

《微信小程序开发从入门到实战》学习二十六

3.4 开发参与投票页面 参与投票页面同样需要收集用户提交的信息&#xff0c;哪个用户在哪个投票选择了什么选项&#xff0c;因此它也是一个表单页面 3.4.1 如何获取投票信息 假设用户A在投票创建页面后填了表单&#xff08;1.创建投票&#xff09;&#xff0c;用户A 点了提交…

Antd Design的inputNumber实现千位分隔符和小数点并存

代码来自文章: react中使用antDesign的Input/InputNumber最多保留两位小数&#xff0c;多的小数位禁止输入&#xff0c;且实现输入实时校验并添加千位分隔符, 正则忘了很多, 我主要做个笔记. //定义InputNumber的参数 const NumberProps {min: 0,//最小值max: …

SQLite3

数据库简介 常用的数据库 大型数据库&#xff1a;Oracle 中型数据库&#xff1a;Server 是微软开发的数据库产品&#xff0c;主要支持 windows 平台。 小型数据库&#xff1a;mySQL 是一个小型关系型数据库管理系统&#xff0c;开放源码 。(嵌入式不需要存储太多数据。) SQL…

spark shuffle 剖析

ShuffleExchangeExec private lazy val writeMetrics SQLShuffleWriteMetricsReporter.createShuffleWriteMetrics(sparkContext)private[sql] lazy val readMetrics SQLShuffleReadMetricsReporter.createShuffleReadMetrics(sparkContext)用在了两个地方&#xff0c;承接的是…

HarmonyOS ArkTS 应用添加弹窗(八)

概述 在我们日常使用应用的时候&#xff0c;可能会进行一些敏感的操作&#xff0c;比如删除联系人&#xff0c;这时候我们给应用添加弹窗来提示用户是否需要执行该操作&#xff0c;如下图所示&#xff1a; 弹窗是一种模态窗口&#xff0c;通常用来展示用户当前需要的或用户必须…

基于金枪鱼群算法优化概率神经网络PNN的分类预测 - 附代码

基于金枪鱼群算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于金枪鱼群算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于金枪鱼群优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要&#xff1a;针对PNN神…

oracle “ORA-25153:临时表空间为空”

从生产上面备份出来了一个数据库&#xff0c;应用在使用时显示ORA-25153临时表空间为空的报错&#xff0c;原因一般是数据库迁移时&#xff0c;没有迁移完整造成的 解决方法 1.创建新的临时表空间temp2 create temporary tablespace temp2 tempfile DATA size 100M autoexten…

通过AX6000路由器,实现外部访问内网的任意主机

概述 这里遇到一个场景,就是需要外部的人员,访问我内网的一台设备,进行内外部的设备联调。 这也是实际环境中,很常见的一种场景。 之前的做法是子设备上运行edge节点,可以直接访问。 但有的设备无法运行edge节点,那么可以参考一下这个方案来实现。 此方案可以摒弃了…

OpenAI再次与Sam Altman谈判;ChatGPT Voice正式上线

11月22日&#xff0c;金融时报消息&#xff0c;OpenAI迫于超过700名员工联名信的压力&#xff0c;再次启动了与Sam Altman的谈判&#xff0c;希望他回归董事会。 在Sam确定加入微软后&#xff0c;OpenAI超700名员工签署了一封联名信&#xff0c;要求Sam和Greg Brockman&#x…

从0开始学习JavaScript--JavaScript迭代器

JavaScript迭代器&#xff08;Iterator&#xff09;是一种强大的编程工具&#xff0c;它提供了一种统一的方式来遍历不同数据结构中的元素。本文将深入探讨JavaScript迭代器的基本概念、用法&#xff0c;并通过丰富的示例代码展示其在实际应用中的灵活性和强大功能。 迭代器的…

本地训练,开箱可用,Bert-VITS2 V2.0.2版本本地基于现有数据集训练(原神刻晴)

按照固有思维方式&#xff0c;深度学习的训练环节应该在云端&#xff0c;毕竟本地硬件条件有限。但事实上&#xff0c;在语音识别和自然语言处理层面&#xff0c;即使相对较少的数据量也可以训练出高性能的模型&#xff0c;对于预算有限的同学们来说&#xff0c;也没必要花冤枉…

如何用惯性动作捕捉系统,快速创建数字人三维动画?

在动画制作领域&#xff0c;惯性动作捕捉技术已经逐渐成为一种重要的制作手段。通过动捕设备能够将动捕演员真实的动作转化为数字数据&#xff0c;然后在动画中再现这些动作。为了创造出逼真、流畅的数字人动画&#xff0c;惯性动作捕捉系统成为了一大工具。 根据采集方式的不…

【计算机网络笔记】路由算法之层次路由

系列文章目录 什么是计算机网络&#xff1f; 什么是网络协议&#xff1f; 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能&#xff08;1&#xff09;——速率、带宽、延迟 计算机网络性能&#xff08;2&#xff09;…

2023年亚太地区数学建模大赛 问题B

玻璃温室中的微气候法规 温室作物的产量受到各种气候因素的影响&#xff0c;包括温度、湿度和风速[1]。其中&#xff0c;适宜的温度和风速是植物生长[2]的关键。为了调节玻璃温室内的温度、风速等气候因素&#xff0c;温室的设计通常采用带有温室风扇的通风系统&#xff0c;如…

《数学之美》第三版的读书笔记一、主要是马尔可夫假设、隐马尔可夫模型、图论深度/广度、PageRank相关算法、TF-IDF词频算法

1、马尔可夫假设 从19世纪到20世纪初,俄国有个数学家叫马尔可夫他提出了一种方法,假设任意一个词出现的概率只同它前面的词有关。这种假设在数学上称为马尔可夫假设。 2、二元组的相对频度 利用条件概率的公式,某个句子出现的概率等于每一个词出现的条件概率相乘,于是可展…

2023亚太杯数学建模B题思路 - 玻璃温室中的微气候法规

# 1 赛题 问题B 玻璃温室中的微气候法规 温室作物的产量受到各种气候因素的影响&#xff0c;包括温度、湿度和风速[1]。其中&#xff0c;适 宜的温度和风速是植物生长[2]的关键。为了调节玻璃温室内的温度、风速等气候因素 , 温室的设计通常采用带有温室风扇的通风系统&#x…