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

前言:Linux进程控制包含了进程终止,进程等待,进程程序替换。走到现在我们也只剩下进程程序替换没介绍了,那么让我们来看看进程程序替换到底是什么!


在这里插入图片描述


本篇主要内容:
替换原理
替换函数
实现简易shell

在这里插入图片描述


我们所创建的所有的子进程,执行的代码,都是父进程代码的一部分,如果我们想让子进程执行新的程序,执行全新的代码和访问全新的数据,不在和父进程有瓜葛这就要用到我们的进程替换了!!!


进程程序替换

  • 1. 替换原理
  • 2. 替换函数
    • 2.1 execl类型函数
    • 2.2 execv类型函数
  • 3. 实现简易shell
  • 4. 总结


1. 替换原理

我们先介绍一个exec函数,方便我们读下面的代码
execl:
在这里插入图片描述
我们直接来看代码分析:

  1 #include <stdio.h>2 #include <sys/types.h>                                                                                                                                                                 3 #include <unistd.h>4 #include <wait.h>5 int main()6 {7     pid_t id = fork();8     if(id == 0)9     {10         printf("pid: %d, exec command begin\n", getpid());11         sleep(2);12         execl("/usr/bin/ls", "ls", "-a", "-l", NULL);13         printf("pid: %d, exec command end\n", getpid());14     }15     else{16         pid_t rid = waitpid(-1, NULL, 0);17         if(rid > 0)18         {19             printf("wait success, pid: %d\n", rid);20         }21     }22     return 0;23 }

在这里插入图片描述

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

  • 操作系统在调用exec函数时,只要将整个进程的代码和数据用新程序的代码和数据进行替换。
  • 子进程往往会共享父进程的代码和数据,因为有写时拷贝的技术,有新数据写入时,会重新开辟空间所以不会相互影响,保证父子进程的独立性。

在这里插入图片描述


但是我们仔细查看代码和执行结果后发现,竟然有一段代码没有执行!

13  printf("pid: %d, exec command end\n", getpid());

其实是调用exec程序替换,只要exec替换新程序成功了,子进程就回去执行新的程序,因此后面的代码将不会被执行


2. 替换函数

在进程程序替换中,替换函数必不可少,让我们来看看这类函数!

在这里插入图片描述

RETURN VALUE:
The exec() functions error has occurred return only if an The return value is -1 and errno to indicate the error

而exec开头的函数一共介绍6种,它们只会在出错时返回 -1!

我们以一下两点来理解:

  • 必须先找到这个可执行程序怎么执行
  • 必须告诉exec,怎么执行*

2.1 execl类型函数

在这里插入图片描述
在这里插入图片描述

execl我们在上面简单了解了,我们就介绍一下剩下两个

execlp:
在这里插入图片描述

int main()
{{printf("pid: %d, exec command begin\n", getpid());sleep(2);execlp("ls", "ls", "-a", "-l", NULL);  //第一个ls代表怎么找到这个程序 //第二个ls其实是命令的一部分,对应怎么执行                                                                                                                                    printf("pid: %d, exec command end\n", getpid());}                                              return 0;                          
}  

在这里插入图片描述

在使用execlp函数时,我们并没有加上路径,但是我们依然能够获得想要的结果,正是因为p代表了PATH他会自动去环境变量PATH中根据file来寻找可执行程序!!!


execlpe:
在这里插入图片描述

// mytest源代码
int main()
{char *const myenv[] = {"myenv1=2004","myenv2=2005","myenv3=2005","myenv4=2005",NULL};                                                                                                                                                                               printf("exec command begin\n");execle("./myprocess", "myprocess", NULL, myenv);// myprocess源代码
int main(int agrc, char *agrv[], char *env[])
{for(int i = 0; env[i]; i++){printf("[%d]: %s\n", i, env[i]);}return 0;
} 

在这里插入图片描述
这是exec*函数程序替换,我们在下文会讲


2.2 execv类型函数

execv其实就是在execl的基础上多了一个指针数组,情况大差不差我们就不具体介绍了!具体关系如下图:
在这里插入图片描述

在这里插入图片描述
execv 和 execvp:
在这里插入图片描述

int main()
{char *const argv[] = {"ls","-a","-l",NULL};printf("pid: %d, exec command begin\n", getpid());sleep(2);execv("/usr/bin/ls", argv);       //execvp("ls", argv);                                                                                                                         printf("pid: %d, exec command end\n", getpid());  return 0;                          
}  

在这里插入图片描述

其实exec这类函数命令有一定的规律:

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

在这里插入图片描述


3. 实现简易shell

我们用程序替换是可以用C语言把C++调用起来的

// test.c
int main()
{printf("pid: %d, exec command begin\n", getpid());execl("./myprocess", "myprocess", NULL);                                                                                                                               return 0;
}// test.cc
int main()
{std::cout << "hello c++" << std::endl;std::cout << "hello c++" << std::endl;std::cout << "hello c++" << std::endl;std::cout << "hello c++" << std::endl;                                                                                                                                             return 0;
}

Makefile:

.PHONY:allall:mytest myprocessmyprocess:test.ccg++ -o $@ $^ -std=c++11mytest:test.cgcc -o $@ $^ -std=c99.PHONY:clean
clean:rm -f mytest myprocess 

因为我们要同时编译两个程序,所以我们要对Makefile进行一下调整

在这里插入图片描述

我们通过程序替换将C++用C语言调用起来了


那我们除了调用C++还能调用别的嘛?

  • 显而易见当然是可以的!!!
// test.sh
// 会简单编写就好
#!/usr/bin/bash
echo "hello Linux"
echo "hello world"
touch code// test.c
int main()
{printf("pid: %d, exec command begin\n", getpid());execl("/usr/bin/bash", "bash", "test.sh", NULL);                                                                                                                               return 0;
}

在这里插入图片描述

当然除了我们讲的这两个,其实所有语言都能通过进程程序替换来调用,因为我们用任意语言写的程序都会变成进程,只要是进程就都可以用exec*来替换——系统大于一切


4. 总结

进程程序替换能够让我们只用C语言就可以调用其他任何语言去执行,极大程度上带来了便利,也可以帮助我们完成很多任务,好了,关于Linux进程控制我们先了解到这,下次我们将进入新的章节!

谢谢大家支持本篇到这里就结束了
在这里插入图片描述

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

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

相关文章

2024年最新软件测试面试题必问的1000题!

我了解的测试理论和方法包括以下几个方面&#xff1a; 黑盒测试与白盒测试&#xff1a; 黑盒测试&#xff1a;基于对软件系统外部行为进行测试&#xff0c;独立于内部代码实现细节。黑盒测试关注输入与输出之间的关系以及软件功能是否符合预期。白盒测试&#xff1a;基于对软件…

【一步一步了解Java系列】:了解Java与C语言的运算符的“大同小异”

看到这句话的时候证明&#xff1a;此刻你我都在努力~ 加油陌生人~ 个人主页&#xff1a; Gu Gu Study ​​ 专栏&#xff1a;一步一步了解Java 喜欢的一句话&#xff1a; 常常会回顾努力的自己&#xff0c;所以要为自己的努…

外卖系统拦截器实现(Interceptor)

SpringMVC的拦截器主要是用于拦截控制器方法的执行&#xff1b; 概念&#xff1a;是一种动态拦截方法调用的机制&#xff0c;类似于过滤器。在Spring中动态拦截控制器中方法的执行。 作用&#xff1a;在指定的控制器中调用前后执行预先设定的代码&#xff0c;完成功能增强。 应…

如何查看SNMP设备的OID

什么是OID和MIB OID OID 代表对象标识符。 OID 唯一地标识 MIB 层次结构中的托管对象。 这可以被描述为一棵树&#xff0c;其级别由不同的组织分配。MIB MIB&#xff08;管理信息基&#xff09;提供数字化OID到可读文本的映射。 使用MIB Browser扫描OID 我的设备是一台UPS SN…

四、基于Stage模型的应用架构设计

前面我们了解了如何构建鸿蒙应用以及开发了第一个页面&#xff0c;这只是简单的demo&#xff1b;那么如何去设计&#xff0c;从0到1搭建一个真正的应用呢 一、基本概念 1、Stage模型基本概念 Stage模型概念图 AbilityStage&#xff1a;是一个Module级别的组件容器&#xff0…

数学建模——农村公交与异构无人机协同配送优化

目录 1.题目 2.问题1 1. 问题建模 输入数据 ​编辑 2. 算法选择 3.数据导入 3.模型构建 1. 距离计算 2. 优化模型 具体步骤 进一步优化 1. 重新定义问题 2. 变量定义 3. 优化目标 具体步骤 再进一步优化 具体实现步骤 1. 计算距离矩阵 2. 变量定义 3. 约束…

前端基础知识-ES6扩展运算符(快速实现数组添加新元素、字符串转为数组、对象添加新属性)

前言&#xff1a; 扩展运算符又称为Rest运算符&#xff0c;可以实现数组、对象、字符串在语法层面上的展开&#xff0c;达到简化语法的目的&#xff0c;使得我们提高开发效率 主要用法&#xff1a; 在需要解构的变量前加三个点&#xff08;...xxx&#xff09; 具体示例&…

ubuntu CUDA 驱动更新,版本更新,多CUDA版本管理

1 新版本驱动下载 前面介绍过window CUDA驱动更新&#xff0c;但是对于ubuntu 的驱动更新&#xff0c;没有一键操作。 本人笔记本电脑n年前装的CUDA DRIVER仅支持到cuda10.2&#xff0c;实在无法满足这日新月异的科技更新。 左 旧的驱动版本 右 新下载的硬件支持的驱动版本&…

es 分词器(五)之elasticsearch-analysis-jieba 8.7.0

es 分词器&#xff08;五&#xff09;之elasticsearch-analysis-jieba 8.7.0 今天咱们就来讲一下es jieba 8.7.0 分词器的实现&#xff0c;以及8.x其它版本的实现方式&#xff0c;如果想直接使用es 结巴8.x版本&#xff0c;请直接修改pom文件的elasticsearch.version版本号即可…

那些年我与c++的叫板(一)--string类自实现

引子&#xff1a;我们学习了c中的string类&#xff0c;那我们能不能像以前数据结构一样自己实现string类呢&#xff1f;以下是cplusplus下的string类&#xff0c;我们参考参考&#xff01; 废话不多说&#xff0c;直接代码实现&#xff1a;&#xff08;注意函数之间的复用&…

Shell之(数组)

目录 一、shell数组 1.数组的定义 2.定义数组的方法 第一种 第二种 第三种 第四种 3.数组分片 4. 数组字符替换 临时替换 永久替换 5.删除数组 删除指定的下标 删除整组 6.数组遍历和重新定义 7.数组追加元素 方式一&#xff1a;指定位置添加 方法二&a…

AnyMP4 Video Converter for Mac/Win - 视频转换的卓越之选

在当今数字化的时代&#xff0c;视频内容无处不在&#xff0c;而拥有一款强大的视频转换器就显得至关重要。AnyMP4 Video Converter for Mac/win 正是这样一款出类拔萃的工具&#xff0c;为您带来高效、便捷的视频转换体验。 这款视频转换器具备令人惊叹的功能。它支持广泛的视…