Linux控制---进程程序替换

       前言:前面我们学洗了Linux进程退出的相关知识,了解了什么是进程退出,已经进程等待的相关话题,今天,我们来学习Linux中的进程程序替换,进程程序替换在Linux中可以用于实现新程序的启动、程序升级、多进程程序和程序恢复等多种应用。

目录

1、进程程序替换

什么是程序替换?

替换原理

单个进程的程序替换(无父子进程)

execl函数

程序替换原理详述

父子进程中的程序替换

替换进程如何得知要从程序的最开始开始执行?又如何得知最开始的地方在哪里?

替换程序覆盖性

exec函数分类

execlp、execv、execvp函数

程序替换只能替换系统程序?

C语言替换C++或python程序

多文件同时编译的自动化makefile实现

替换自己写的程序

c语言种替换shell脚本语言

环境变量与execle、execvpe函数

定义子进程自己的专属环境变量

execve系统调用


1、进程程序替换

什么是程序替换?

       在我们之前的讲解中,父子进程本质上都在使用同一份代码,这也就相当于父子进程在同一套代码之下完成任务,而我们创建子进程的目的本来就是为了能让子进程完成一些父进程需要其完成的任务,并带给父进程对应的返回结果,但是,如果父子进程在同一套代码逻辑中,那创建子进程不就相当于脱裤子放屁自找麻烦吗?其实,子进程被创建出来,是可以用来执行与父进程不同的程序的,程序替换简单来说就是将正在运行的程序替换为另一个程序。这个过程涉及内存加载新程序到内存中,并更新页表信息,初始化虚拟地址空间,然后将现有PCB的指针指向新的程序,使得现有PCB从头开始调度新的进程运行。需要注意的是,程序替换后,当前进程运行完替换后的程序就会退出,并不会回去运行原先的程序。

替换原理

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

单个进程的程序替换(无父子进程)

execl函数

      execl函数时exec函数中的一种,后面的几种我们稍后会提及,其功能替换一个指定路径的程序,

其中execl的参数,path参数就是替换程序所在的文件路径,arg就是执行命令的若干个参数,比如ls -a -l......,这些都属于这个参数,这里,我们先给出一个最简单的解释,比如我们需要执行“ls -a -l”,那么我们在命令行参数中就按这个顺序输入(“ls”,“-a”,“-l”)就可以了,也就相当于我们在命令行中如何使用,在参数中就如何输入即可,需要注意的是,该函数必须在参数最后加上NULL表示传入参数结束。具体如下面的用例:

我们发现一个现象,一旦我们进行了程序替换,也就是执行了execl函数,该函数之后的语句就不会再被执行了,上面的例子中,execl函数之后的printf函数没能够被执行,这一点就和我们程序替换的原理有关,下面我们就来对程序替换原理进行详细的分析,

程序替换原理详述

       在Linux中,进程程序替换通常指的是使用exec()系列函数来实现的进程替换过程。这个过程包括加载一个全新的程序(代码和数据)到调用进程的地址空间中,并开始执行新程序。这个过程会替换当前进程的内容,原有进程的执行状态和上下文信息会被丢弃。在进程替换发生时,原有进程的代码、数据和资源会被新的进程所取代。该替换过程没有创建新的进程,只是替换原来的进程

父子进程中的程序替换

       前面我们演示的是单进程的程序替换场景,如果在父子进程的模式下又会出现什么样的情况呢?我们知道,父子进程在不进行写操作时是共享同一份代码和数据的,但是如果父进程或者子进程发生了程序替换,那么此时剩下的那个进程就要采用我们之前学过的写时拷贝技术来将原有代码和数据拷贝一份供剩余进程使用,使得两个进程各自保持自己的独立性

替换进程如何得知要从程序的最开始开始执行?又如何得知最开始的地方在哪里?

       当我们采用程序替换的方式引入一个新的程序时,这个程序如何得知其该从哪个位置开始执行,又如何得知其执行的位置在哪里呢?实际上,每个可执行程序在磁盘中的存储,并不是单纯的只是存储代码和数据,可执行程序的二进制文件中,头部有一个专门的表结构,其中专门有一个字段用来表示程序的入口地址,通过将其填充到对应的调用进程的程序计数器pc中,即可达到调用的效果。这里顺便提一嘴,pc指针(eip)是cpu中的一个寄存器,在cpu中只有一个,但是并不是所有的进程只有一个pc指针,寄存器中的内容可以有多份,一个cpu可以有多套内容,分别用来对应不同的内容。

替换程序覆盖性

        我们在上面的代码中不难发现,不论是单个进程的程序替换,还是父子进程的程序替换,被替换的进程在执行exec函数结束后,其后续的代码均不会被执行,此时因为程序被替换,所以被替换程序会被覆盖,导致其后序的代码不会再被执行,这个就是替换程序的覆盖性。

exec函数分类

       前面我们了解到exec函数具有多种,但是各种函数之间的用法和功能却不尽相同,下面我们来介绍几种常见的exec函数:

execlp、execv、execvp函数

程序替换只能替换系统程序?

       上面我们的例子只是针对系统程序进行了替换,但是不代表程序替换只能局限于系统函数,事实上,程序替换还可以替换自己写的程序,这其中包括其他语言的程序(例如c++和python程序)还有shell脚本语言等,下面,我们就来演示一下。

C语言替换C++或python程序

首先,我们来分别编写一个C++和python文件:

多文件同时编译的自动化makefile实现

      当我们想要一次编译两个或两个以上的源文件并形成对应的目标文件时,我们就需要只调用一次makefile工具就能直接将所有源文件全部编译,但是我们知道makefile从上往下扫描的时候默认只形成扫描到的第一个目标文件,所以我们就需要采取一些手段,使其能够同时编译出两份目标文件,我们可以将两份目标文件封装成一个总的目标文件放在makefile首部,这样,默认调用的首个目标文件中就包含了我们需要的两个目标文件,从而达到目的。

比如,我们一次需要形成C和C++的可执行程序时,我们可以采用如下的方式编写makefile:

替换自己写的程序

python程序也是同理,python是一种解释型语言,Python 解释器在运行代码时不会将其编译为机器语言。相反,它会在运行时解释代码,所以,我们不能够在通过makfile编译形成目标文件来形成所谓的“python的可执行程序”了,我们只能通过调用python的解释器来直接解释我们的python文件,具体实现如下:

c语言种替换shell脚本语言

       脚本语言,或称为动态编程语言,是一种解释型语言。它通常用于执行具有复杂逻辑和流程控制的任务,比如网页编程、系统配置、游戏脚本等。其采用系统的bash命令行解释器就可以运行,比如我们可以编写和运行如下的shell脚本,至于如何编写这里先了解即可,这不是我们的重点:

此时,我们可以将其替换为在程序中运行,如下:

      现在,你可以拿着这个和你的朋友装一下了,我们可以在c上运行其他语言的代码,其实不仅是c语言,其他的编程语言都有着类似的程序替换函数,感兴趣的可以去百度一下。

环境变量与execle、execvpe函数

       环境变量一般是具有全局属性的,也就是说,它可以被进程和子进程之间共享,对于父子进程之间,环境变量共用一套无可厚非,并且,我们也知道,父进程的环境变量来源于bash命令行解释器,为了验证bash、父进程和子进程的爷、父、孙的关系,我们可以将父进程中子进程进程替换的代码修改为输出打印环境变量,接着我们利用export向命令行bash进程中加入一个新的临时环境,看其是否会被父进程和子进程所接受和显示,参考代码如下:

接着,我们在命令行导入一个临时环境变量到bash进程中去:

此时,该环境变量就被保存在了bash进程中,接着,我们编译并运行代码,然后,我们可以发现打印出的孙子进程的环境变量中存在我们创建的环境变量(也就是来自“爷爷”进程的环境变量)。

        事实上,父子进程在传递的过程中,子进程是默认继承父进程的去全部的环境变量的,并且不会受到程序替换的影响,前面已经提到,程序替换本质上并没有改变进程的pcb和进程地址空间等内核数据结构,只是替换了物理内存上的代码和数据而已,而环境变量则是随着进程地址空间上的一个字段上保存的,所以,子进程在创建的时候会继承父进程的进程地址空间,也就把父进程的环境变量给继承了下来。

定义子进程自己的专属环境变量

     如果我们的子进程想要定义属于自己的环境变量而不希望再沿用父进程的环境变量,就需要用到我们下面的两个函数了:

例如,此时我们想要修改子进程的被替换程序的环境变量,参考代码如下:

为了方便显示出我们传入命令行参数的效果,我们可以在替换程序中将命令行参数打印出来: 

运行代码,结果如下:

       可以看到,被替换程序的环境变量已经被覆盖式的修改成了我们自己定义的环境变量,注意是覆盖不是新增,那当我们需要新增时怎么办?还记得我们上面给bash添加环境变量时的场景吗?想要给子进程在原来的基础上新增环境变量,我们只需要在其父进程里创建一个临时环境变量就可以了,此时子进程将会继承父进程的环境变量表,包括新增的。

execve系统调用

       execve接口是在Unix和类Unix操作系统中执行新程序的系统调用。事实上,只有execve是真正的系统调用,其它六个函数最终都调用execve,所以execve在man手册 第2节,其它函数在man手册第3节。这些函数之间的关系如下图所示。

本篇到这里就结束了,下期,我们将自己动手做一个简单的shell,可以实现简易的指令反馈和操作,模拟在XShell上的各种指令,持续更新......

       真正能给你撑腰的是丰富的知识储备、足够的经济基础,持续稳定的情绪,可控的生活节奏,以后的日子里,希望你多长本事,多看世界,多走些路,把时间花在正事上,变成自己打心底里喜欢的人,世界上任何东西都应该是被吸引来的,而不是强求来的,梧高峰必至,花香蝶自来。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【数据结构高阶】二叉搜索树

接下来我们来开始使用C来详细讲解数据结构的一些高阶的知识点 本期讲解的是二叉搜索树,对于初阶二叉树有所遗忘的同学可以看到这里: 【精选】【数据结构初阶】链式二叉树的解析及一些基本操作 讲解二叉搜索树主要是为了后面的map和set做铺垫&#xff…

我记不住的getopt_long的那些参数和返回值

前言:最近在学习面向Linux系统进行C语言的编程,通过查询man手册和查看网络上的各种文章来获取一点点的知识,重点是看完手册还是一脸懵逼,搞不懂手册里面再说啥,而本篇文章将记录一下学习getopt_long的那些参数和返回值…

ElasticStack日志分析平台-ES 集群、Kibana与Kafka

一、Elasticsearch 1、介绍: Elasticsearch 是一个开源的分布式搜索和分析引擎,Logstash 和 Beats 收集的数据可以存储在 Elasticsearch 中进行搜索和分析。 Elasticsearch为所有类型的数据提供近乎实时的搜索和分析:一旦数据被索引&#…

0基础学习PyFlink——水位线(watermark)触发计算

在《0基础学习PyFlink——个数滚动窗口(Tumbling Count Windows)》和《0基础学习PyFlink——个数滑动窗口(Sliding Count Windows)》中,我们发现如果窗口中元素个数没有把窗口填满,则不会触发计算。 为了解决长期不计算的问题&a…

PyCharm 【unsupported Python 3.1】

PyCharm2020.1版本,当添加虚拟环境发生异常: 原因:Pycharm版本低了!不支持配置的虚拟环境版本 解决:下载PyCharm2021.1版本,进行配置成功!

2023年,全球CIO最关注的问题是什么?

面对AI大潮,全球CIO们在焦虑什么?随着全球数字化转型步伐的加速,CIO的角色发生了哪些转变? 继2022年5月发布首份全球CIO报告之后,联想集团今年又发布了以“韧性的全球首席信息官(The Resilient CIO&#xf…

python大数据毕设选题

文章目录 0 前言1 大数据毕设选题推荐2 开题指导3 最后 0 前言 大家好!大四的同学们,毕业设计的时间即将到来,你们准备好了吗?为了帮助大家更好地开始毕设,我作为学长给大家整理了最新的计算机大数据专业的毕设选题。…

微信公众号与小程序打通:流量变现的新路径

随着移动互联网的迅速发展,微信公众号和小程序已经成为企业营销和运营的重要工具。将微信公众号与小程序打通,不仅可以提高用户体验,还能有效提升流量的变现效率。本文将为您解析如何打通微信公众号与小程序,让流量快速变现。 一、…

开发知识点-Git

团队协作-Git Giteegitee 创建仓库打开项目所在目录,右键选择Git Bush Here(你要确定电脑上已经安装了Git)初始化本地仓库配置验证信息。 完美解决github访问速度慢介绍Git 与 SVN 区别IDEA 添加 gitee Gitee Git Gitee 大家都知道国内访问 Github 速度…

【前段基础入门之】=>CSS3新特性 响应式布局

文章目录 概念媒体查询媒体类型媒体特性媒体运算符 概念 所谓对响应式布局方案的理解,众说纷纭,核心点就是同一套代码在不同尺度屏幕下的布局呈现方式的不同 社区中有很多人分享,并列出了多种实现响应式布局的方案,比如【 rem&…

quickapp_快应用_快应用组件

快应用组件 web组件web页面与快应用页面通信网页接收/发送消息网页接收消息 快应用页面接收/发送消息给网页发送消息 通信前提- trustedurl list组件refresh组件语法error-使用refresh组件会改变页面高度!refresh组件list组件实现下拉刷新 tab组件 web组件 作用&am…

微信抽奖活动怎么做

微信抽奖活动:打破传统,创新互动,带给你超乎想象的惊喜体验! 随着互联网的飞速发展,人们越来越热衷于参与各种线上活动。而微信,作为中国最大的社交平台之一,自然成为了各种活动的聚集地。今天…

IntelliJ IDEA 安装 GitHub Copilot插件 (最新)

注意: GitHub Copilot 插件对IDEA最低版本要求是2021.2,建议直接用2023.3,一次到位反正后续要升级的。 各个版本的依赖关系,请参照: ##在线安装: 打开 IntelliJ IDEA扩展商店,输入 "Git…

IntelliJ IDEA启动一个普通的java web项目的配置

原创/朱季谦 这是我很久以前刚开始用IntelliJ IDEA时记录的笔记,应该是五年前的一篇笔记了。正好赶上最近离职了,可以有比较多的时间把以前的记录整理一下,可以让刚接触到IntelliJ IDEA的童鞋学习如何在IntelliJ IDEA引入一个单机版的jar形式…

【python零基础入门学习】python进阶篇之数据库连接-PyMysql-全都是干货-一起来学习吧!!!

本站以分享各种运维经验和运维所需要的技能为主 《python零基础入门》:python零基础入门学习 《python运维脚本》: python运维脚本实践 《shell》:shell学习 《terraform》持续更新中:terraform_Aws学习零基础入门到最佳实战 《k8…

【npm 错误】:npm ERR! code ERESOLVE、npm ERR! ERESOLVE could not resolve问题

用过npm的小伙伴都会有这么一个情况出现,就是npm install /npm install xxxx 会出现改一连串的错误,如下: 解决办法: 只要在npm install后面加上--legacy-peer-deps就可以解决问题,安装插件也一样 npm install --legacy-peer-dep…

原论文一比一复现 | 更换 RT-DETR 主干网络为 【VGG13】【VGG16】【VGG19】| 对比实验必备

本专栏内容均为博主独家全网首发,未经授权,任何形式的复制、转载、洗稿或传播行为均属违法侵权行为,一经发现将采取法律手段维护合法权益。我们对所有未经授权传播行为保留追究责任的权利。请尊重原创,支持创作者的努力,共同维护网络知识产权。 论文地址:https://arxiv.o…

V10服务器安装virt-manage

kvm是什么 KVM(Kernel-based Virtual Machine, 即内核级虚拟机) 是一个开源的系统虚拟化模块。它使用Linux自身的调度器进行管理,所以相对于Xen,其核心源码很少。目前KVM已成为学术界的主流VMM之一,它包含一个为处理器提供底层虚拟化 可加载…

[CISCN 2023 华北]pysym

源码如下 from flask import Flask, render_template, request, send_from_directory import os import random import string app Flask(__name__) app.config[UPLOAD_FOLDER]uploads app.route(/, methods[GET]) def index():return render_template(index.html) app.route…

IDEA写mybatis程序,java.io.IOException:Could not find resource mybatis-config.xml

找不到mybatis-config.xml 尝试maven idea:module&#xff0c;不是模块构造问题 尝试检验pom.xml&#xff0c;在编译模块添加了解析resources内容依旧不行 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.or…