七、进程程序替换

news/2025/2/19 21:39:15/

文章目录

  • 一、进程程序替换
    • (一)概念
    • (二)为什么程序替换
    • (三)程序替换的原理
    • (四)如何进行程序替换
      • 1. execl
      • 2. 引入进程创建——子进程执行程序替换,会不会影响父进程呢?
    • (五)大量的测试各种不同的接口
      • 1.命名理解 (带v和带l的)
      • 2.记忆技巧
      • 3.带e和带p
    • (六)具体接口说明
      • 1.execv
      • 2.execlp
      • 3.execvp
      • 4.execle
  • 二、模拟实现shell
  • 三、内建命令——以chdir为例

一、进程程序替换

(一)概念

子进程执行的是父进程的代码片段,如果我们想让创建出来的子进程,执行全新的程序呢?

需要用到:进程的程序替换!

(二)为什么程序替换

我们一般在服务器设计(linux编程)的时候,往往需要子进程干两件种类事情!

  1. 让子进程执行父进程的代码片段(服务器代码)
  2. 让子进程执行磁盘中一个全新的程序(shell,想让客户端执行对应的程序,通过我们的进程,执行其他人写的进程代码等等),c/c++ ->c/c++/Python/Shell/Php/Java…

(三)程序替换的原理

  1. 将磁盘中的程序,加载入内存结构
  2. 重新建立页表映射,谁执行程序替换,就重新建立谁的映射(子进程)

效果:让我们的父进程和子进程彻底分离,并让子进程执行一个全新的程序!

在这里插入图片描述

这个过程有没有创建新的进程呢?

没有,子进程的PCB等结构并未改变,只是改变的页表映射关系。

程序替换成功后,运行完新程序,则程序直接退出;程序替换成功后,原进程没有退出,使用原进程运行新程序

我们只能调用接口,为什么呢?

因为这个过程实际上是把数据从一个硬件搬到另一个硬件的操作,这个操作只能由OS操作系统完成

(四)如何进行程序替换

man execl 查看进行程序替换的函数:
我们如果要执行一个全新的程序,我们需要做几件事情呢?

1.程序本质就是一个磁盘上的文件,所以我们需要先找到这个程序在哪里
2.程序可能携带选项进行执行(也可以不携带),然后告诉OS,我要怎么执行这个程序?(要不要带选项)

命令行怎么写(ls -l -a), 这个参数就怎么填"ls",“-l”,“-a”,最后必须是NULL,标识参数传递完毕[如何执行程序的]
00

 #include <unistd.h>extern char **environ;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 execvpe(const char *file, char *const argv[],char *const envp[]);

1. execl

makefile

myexec:myexec.cg++ -o $@ $^ -std=c++11 
.PHONY:clean
clean:rm -f myexec

myexec.c

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<assert.h>
#include<sys/types.h>
#include<sys/wait.h>int main(int argc,char *argv[]) {printf("process is running...\n");pid_t id  = fork();assert(id != -1);if (id == 0) {sleep(1);printf("我是一个进程,我的pid是:%d\n",getpid());// 执行ls -l -a 命令//execl("/usr/bin/ls","ls","-l","-a",NULL); // 这里有两个ls, 重复吗?不重复,一个是告诉系统我要执行谁?一个是告诉系统,我想怎么执行// 执行top命令execl("/usr/bin/top","top",NULL);               printf("我执行完毕了,我的pid是:%d\n",getpid());      // 执行完以上的代码,我们发现一个问题!!// 最后一句代码为什么没有被打印出来呢!!!}return 0;
}

在这里插入图片描述
因为进程一旦替换成功,是将当前进程的代码和数据全部替换了!!!

后面的printf是代码吗??有没有被替换呢??当然,已经早就被替换了!!该代码不存在了!!

所以这个程序替换函数,用不用判断返回值?为什么?

答:不用判断返回值,因为只要成功了,就不会有返回值execl,一旦替换成功,是将当前进程的所有代码和数据全部替换了,execl就直接执行ls命令的代码去了。。而失败的时候,必然会继续向后执行!!最多通过返回值得到什么原因导致的替换失败!

2. 引入进程创建——子进程执行程序替换,会不会影响父进程呢?

子进程执行程序替换,会不会影响父进程呢? ?

不会,因为进程具有独立性。
为什么,如何做到的? ?数据层面发生写时拷贝!当程序替换的时候,我们可以理解成为:代码和数据都发生了写时拷贝完成父子的分离!

(五)大量的测试各种不同的接口

1.命名理解 (带v和带l的)

这些函数原型看起来很容易混,但只要掌握了规律就很好记。l(list) : 表示参数采用列表v(vector) : 参数用数组p(path) : 有p自动搜索环境变量PATHe(env) : 表示自己维护环境变量

2.记忆技巧

execl结尾 l 为list,列表传参——>可变参数包,一个一个传。execv结尾 v 为vector,数组传参——>传的是指针数组。

3.带e和带p

带e的都是可以传环境变量的(execle,execvpe)但是会覆盖系统原有的环境变量,把自己传的环境变量交给进程;
不带e是默认继承系统的环境变量;带p的都是可以自带路径的,直接传命令名称即可(execlp,execvp,execvpe)

(六)具体接口说明

1.execv

int execv(const char *path, char *const argv[]);        

path 依然是程序的路径,参数 argv[] 是存着要实现指令的指针数组
在这里插入图片描述

char *const argv_[] = {"ls","-a","-l","--color=auto",NULL
};

2.execlp

int execlp(const char *file, const char *arg, ...);        带p的就传程序名即可file:要执行的程序。执行指令的时候,默认的搜索路径,在哪里搜索呢?在环境变量PATH
命名带p的,可以不带路径,只说出你要执行哪一个程序即可!execlp("ls""ls", "-a", "-1 "NULL)
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<assert.h>
#include<sys/types.h>
#include<sys/wait.h>int main(int argc,char *argv[]) {printf("process is running...\n");pid_t id  = fork();assert(id != -1);if (id == 0) {sleep(1);printf("我是一个进程,我的pid是:%d\n",getpid());//execl("/usr/bin/ls","ls","-l","-a",NULL); // 这里有两个ls, 重复吗?不重复,一个是告诉系统我要执行谁?一个是告诉系统,我想怎么执行// 执行top命令char *const argv_[]={(char*)"ls",(char*)"-a",(char*)"-l",NULL};//execl("/usr/bin/top","top",NULL);         execlp("ls","ls","-a","-l",NULL);//这里出现了两个ls,含义一样吗?不一样!exit(1);   //execvp("ls", argv_);printf("我执行完毕了,我的pid是:%d\n",getpid());      // 执行完以上的代码,我们发现一个问题!!// 最后一句代码为什么没有被打印出来呢!!!}int status = 0;int ret = waitpid(id,&status,0);if(ret == id){sleep(2);printf("父进程等待成功!\n");}return 0;
}

在这里插入图片描述

3.execvp

 int execvp(const char *file, char *const argv[]);
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<assert.h>
#include<sys/types.h>
#include<sys/wait.h>int main(int argc,char *argv[]) {printf("process is running...\n");pid_t id  = fork();assert(id != -1);if (id == 0) {sleep(1);printf("我是一个进程,我的pid是:%d\n",getpid());//execl("/usr/bin/ls","ls","-l","-a",NULL); // 这里有两个ls, 重复吗?不重复,一个是告诉系统我要执行谁?一个是告诉系统,我想怎么执行// 执行top命令char *const argv_[]={(char*)"ls",(char*)"-a",(char*)"-l",NULL};//execl("/usr/bin/top","top",NULL);         //execlp("ls","ls","-a","-l",NULL);//这里出现了两个ls,含义一样吗?不一样!exit(1);   execvp("ls", argv_);printf("我执行完毕了,我的pid是:%d\n",getpid());      // 执行完以上的代码,我们发现一个问题!!// 最后一句代码为什么没有被打印出来呢!!!}// 父进程int status = 0;int ret = waitpid(id,&status,0);if(ret == id){sleep(2);printf("父进程等待成功!\n");}return 0;
}

在这里插入图片描述

4.execle

int execle(const char *path, const char *arg,..., char * const envp[]);

这里的前几个接口都非常熟悉了,这里最后一个接口叫做环境变量。那么为什么要有这个接口呢?

说到环境变量之前我们先来看一下这个问题,我们刚刚提到过,进程替换可以让我们执行其他语言写的程序,那么我们怎么来执行呢?(我们使用execl 函数来调用)

我们现在的目标是想用我们写的myexec.c把mycmd.cpp调用起来,那么怎么来用呢?
myexec.c

我们当前使用的是绝对路径来调用我的mycmd程序!

当然我们也可以使用相对路径来调用。

相对路径调用——
makefile

.PHONY:all
all: mybin myexec
mybin:mybin.cg++ -o $@ $^ -std=c++11
myexec:myexec.cg++ -o $@ $^ -std=c++11
.PHONY:clean
clean:rm -f myexec mybin
#include<stdio.h>
#include<unistd.h>
#include<sys/wait.h>
#include<stdlib.h>int main()
{printf("我是一个进程,我的pid是:%d\n",getpid());pid_t id=fork(); if(id==0){//childprintf("我是子进程,我的pid是:%d\n",getpid());execl("./mycmd","mycmd",NULL);exit(1);}//一定是父进程int status=0;int ret=waitpid(id,&status,0);if(ret==id){sleep(2);printf("父进程等待成功!\n");}return 0;
}

在这里插入图片描述
为什么会有这么多接口?——因为要适配应用场景。

execve为什么是单独的?——实际上,只有 execve是系统调用,其他都是对系统接口的封装,最后都要调用到execve!
在这里插入图片描述

二、模拟实现shell

三、内建命令——以chdir为例


http://www.ppmy.cn/news/262582.html

相关文章

Flutter的状态管理之Provider

Provider简介 Flutter Provider是Flutter中一个非常流行的状态管理库&#xff0c;它可以帮助开发者更加方便地管理应用程序中的状态。Provider提供了一种简单的方式来共享和管理应用程序中的数据&#xff0c;并且可以根据数据的变化来自动更新UI界面。 Provider的核心思想是将…

linux主机diy,DIY属于自己的Linux系统

本篇博文主要介绍对linux系统的裁剪&#xff0c;所以要熟悉linux的启动过程&#xff0c;linux的启动过程主要有这几个步骤&#xff1a;POST(开机自检)-->BIOS引导设备-->MBR(bootloader)引导程序-->GRUB菜单-->加载kernel-->执行init脚本初始化系统 1、首先准备…

diy服务器个人主机_服务器DIY也可以这么简单

其次&#xff0c;挑选服务器硬件。 服务器CPU 作用&#xff1a;服务器的CPU可以说是整个服务器的大脑&#xff0c;关系到服务器的整体性能&#xff0c;是衡量服务器性能的首要指标。 主流厂商&#xff1a; 目前流行的是Intel和AMD两个厂商的CPU&#xff0c;其它CPU微乎其微。In…

diy服务器个人主机_小型企业及个人用户入门级服务器DIY

小型企业及个人用户入门级服务器DIY 2007-01-16 eNet&Ciweek 2、主板&#xff1a;服务器主板相比普通PC的主板有很大的不同&#xff0c;这些在前面的介绍中已经说明过了。作为DIY服务器的主板&#xff0c;选购的出发点应是“实际”。主板买回来是用的&#xff0c;如果我们事…

diy计算机工作站,程序猿 篇一:迟到的 618 攒机实录:自建巨硬工作站

程序猿 篇一:迟到的 618 攒机实录:自建巨硬工作站 2018-07-25 18:42:53 3点赞 5收藏 2评论 1. 前言 其实 618 之前,本公子就已经把工作站搞起来了,一直懒癌拖着没有写。暑假来了,应届生入职,小朋友们上大学,又到了每年装备更新的季节。所以,忙里偷闲,特地分享一下我的…

diy 文件服务器,文件服务器diy

文件服务器diy 内容精选 换一换 FTP/SFTP连接适用于从线下文件服务器或ECS服务器上迁移文件到OBS或数据库。当前仅支持Linux操作系统的FTP 服务器。连接FTP或SFTP服务器时&#xff0c;他们的连接参数相同&#xff0c;如表1所示。FTP/SFTP连接参数参数名说明取值样例名称连接的名…

diy nas配置推荐2020_廉价NAS方案,手把手教你用淘汰主机安装黑群NAS

大家好,上次写了一篇NAS总论。聊了聊NAS的一些常用功能和硬件的选择。 作为新手小白来说,我们用手头的主机或者购入了一台类似蜗牛星际的廉价NAS主机之后。该如何装入NAS系统呢?这次我们就来动手组建自己第一个NAS系统。 懒人目录: 1、群辉系统介绍 2、安装工具和软件 3、群…

计算机diy市场前景分析,DIY已经走到末路?PC硬件未来何去何从

原标题:DIY已经走到末路?PC硬件未来何去何从 伴随着DIY硬件的出货量逐年递减,PC行业的颓势已经是不争的事实,未来DIY的道路又在何方?PC硬件又将何去何从,且听我来细细分析。 回想20年前,PC硬件爆炸式的增长速度,那个时候热火朝天的电脑城,配置单更是漫天飞,那时候很多…