【Linux】开始学习进程替换吧!

简介: 通过学习进程替换,我们可以体会到多语言混搭的快乐,可以从C语言直接蹦到python ,也可以从c++里运行java代码。是不是很厉害!这是通过调度多个进程的效果,联系我们之前学习的进程,进程控制等概念。我们可以想要运行其他代码可以通过创建子进程来实现,但是这样也肯定是同一种语言,如果想要运行其他语言,那是不是有种方法可以调度一个进程来当做子进程呢???我们开始今天的学习吧!

送给大家一句话:

人生中有些事,你不竭尽所能去做,你永远不知道你自己有多出色。—— 尾田荣一郎《海贼王》

开始学习进程替换吧

1 前言

通过学习进程替换,我们可以体会到多语言混搭的快乐,可以从C语言直接蹦到python ,也可以从c++里运行java代码。是不是很厉害!这是通过调度多个进程的效果,联系我们之前学习的进程,进程控制等概念。我们可以想要运行其他代码可以通过创建子进程来实现,但是这样也肯定是同一种语言,如果想要运行其他语言,那是不是有种方法可以调度一个进程来当做子进程呢???

我们开始今天的学习吧!

2 进程替换

2.1 替换函数

进程替换有六种以exec开头的函数,统称exec函数

#include <unistd.h>`
int execl(const char *path, const char *arg, ...);//...代表可变参数 类似printf
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 *filename, char *const argv[],char* const envp[]);

我们来进行一下使用,来看看:

  1 #include<stdio.h>
  2 #include<unistd.h>
  3 
  4 int main()
  5 {
  6   printf("textexec ... begin!\n");
  7 
  8   execl("/usr/bin/ls","ls","-l","-a",NULL);                                                                                                                                   
  9                                                                                                
 10   printf("textexec ... end!\n");                                                               
 11   return 0;                                                                                    
 12 }  

可以看到执行了ls -l -a命令(最后的打印语句也没有进行),可是我执行的是我们的代码,怎么就运行了ls 这个程序呢?

原因就是:exec*系列函数可以执行起来新的程序,让进程通过exec函数把自己替换为一个全新的进程

2.2 替换原理

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

我们知道 进程 = 内核数据结构 + 代码和数据 ,替换就是用新进程的代码与数据替换之前的代码与数据,注意不改变pid哦!

站在被替换进程的角度:本质就是这个程序别加载到内存里了!!!exec* 就类似一个Linux 上的加载函数。


而且我们不用关心exec*函数的返回值,只要替换成功了,就不会向后运行(也就用不到它的返回值了),只要继续运行那一定就是替换失败了!!!

那如果不想替换掉我们的程序,还想要打开一个新程序呢??? 我们接着向下看

2.3 单进程改为多进程

方法很简单,我们通过fork函数创建一个子进程,让子进程来执行我们的新程序不就可以了吗!

1 #include<stdio.h>  
  2 #include<unistd.h>   
  3 #include<stdlib.h>
  4 #include<sys/types.h>
  5 #include<sys/wait.h>
  6 
  7 
  8 int main()
  9 {
 10   printf("textexec ... begin!\n");
 11   
 12   pid_t id = fork();
 13   if(id == 0)
 14   {
 15     //child
 16     execl("/usr/bin/ls","ls","-l","-a",NULL);
 17     exit(1);
 18   }
 19 
 20   int status = 0;
 21   pid_t rid = waitpid(-1,&status,0);
 22 
 23   if(rid > 0)
 24   {
 25     printf("father wait success,child exit code:%d \n",WEXITSTATUS(status));                                                                                                  
 26   }                                                                                                                     
 27   printf("testexec ... end!\n");                                                                                        
 28                                                                                                                         
 29   return 0;                                                                                                             
 30 }          

这个程序中:我们通过fork创建了一个子进程,然后在子进程中进行进程替换。这样应该就可以运行ls命令还不会影响原本程序。来看效果:

那么这样以后,创建子进程就变得非常有用了。我们可以通过子进程来完成对应任务:

  1. 让子进程完成父进程代码的一部分
  2. 让子进程执行一个全新的程序(会发生写时拷贝,建立全新的物理内存空间)

2.4 理解使用exec* 函数

这些函数原型看起来很容易混,但只要掌握了规律就很好记。

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

int execl (const char *path, const char *arg, …)

l 表示列表 list

两个参数分别代表:

  1. path (你想执行谁) : 我们执行的程序需要带路径
  2. arg(你想怎么执行) : 咳传多个 , 命令行里怎么运行就怎么写入!!! 举个例子:ls -a -l 所以我们传execl(“/usr/bin/ls” , “ls”,“-a”,“-l”) 。这样就像命令列表

通过这些函数也可以进行执行我们缩写的程序!!!


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

v 表示数组 vector

这个方法就是把列表整合为一个数组,这样就直接传入一个数组(必须以NULL结尾!!!)就可以了:

char* const argv[] = 
{
  (char*)"ls",
  (char*)"-a",
  (char*)"-l",
  (char*)"--color",
  NULL
};
execv("/usr/bin/ls",argv);

是不是很想之前了解过的main函数参数!

int execvp(const char *file, char *const argv[])

带p含义是可以不传入文件路径,可以直接告诉exec*需要执行谁就可以了

本质就是:查找这个程序,系统会在系统环境变量PATH中的路径来寻找

char* const argv[] = 
{
  (char*)"ls",
  (char*)"-a",
  (char*)"-l",
  (char*)"--color",
  NULL
};
execvp("ls",argv);

这个 int execlp(const char *file, const char *arg, …) 同理!!!

execlp("ls" , "ls","-a","-l")

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

这个就就加上了环境变量,让我们支持自主传入

int execve(const char *filename, char const argv[],char const envp[]) 同理 !!!

int execvpe(const char *filename, char const argv[],char const envp[])

这让我们可以传入环境变量给子进程!!!当然也会有一批来着"爷爷进程"bash的环境变量!

envp的含义是全体替换环境变量,所以会有以下情况:

  1. 用全新的给子进程
  2. 用老的环境变量给子进程 char** environ
  3. 老的环境变量稍微修改传给子进程 调用 putenv ()函数
char* const argv[] = 
{
  (char*)"ls",
  (char*)"-a",
  (char*)"-l",
  (char*)"--color",
  NULL
};
//用全新的给子进程
char* const envp[] = 
{
  (char*)"HAHA=111111",
  (char*)"HEHE=222222",
  NULL
};
//putenv("HAHA=111111")
//putenv("HEHE=222222");
//老的环境变量稍微修改传给子进程

execvpe("./myprocess",argv , encp);
//char** environ
// execvpe("./myprocess",argv , environ);

注意

根据上面的用法使用,我们可以总结一下:

函数名 参数格式 是否带路径 是否使用当前环境变量
execl 列表 不是
execlp 列表
execle 列表 不是 不是,需要自己组装环境变量
execv 数组 不是
execvp 数组
execve 数组 不是 不是,需要自己组装环境变量

而且只有 execle 是系统调用!!!他们的关系如下:

Thanks♪(・ω・)ノ谢谢阅读!!!

下一篇文章见!!!

相关文章
|
10月前
|
存储 Linux API
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
在计算机系统的底层架构中,操作系统肩负着资源管理与任务调度的重任。当我们启动各类应用程序时,其背后复杂的运作机制便悄然展开。程序,作为静态的指令集合,如何在系统中实现动态执行?本文带你一探究竟!
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
|
8月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
308 67
|
7月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
225 16
|
7月前
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
144 20
|
6月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
131 0
|
6月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
205 0
|
6月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
133 0
|
6月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
185 0
|
9月前
|
存储 IDE Linux
零基础保姆级教程!手把手教你免费玩转Linux CentOS安装+学习环境搭建(附避坑指南)
本文详细介绍了在VMware虚拟机中安装CentOS 6.8的全过程。首先,需确保已安装VMware并开启V-CPU虚拟化功能,可通过BIOS设置或使用LeoMoon CPU-V工具检测。接着,下载CentOS镜像文件,并在VMware中新建虚拟机,配置CPU、内存、硬盘等参数。最后,加载ISO镜像启动虚拟机,按照提示完成CentOS的安装,包括语言、键盘、存储方式、地区、密码设置及硬盘分区等步骤。安装完成后,以root用户登录即可进入系统桌面,开始学习Linux命令和操作。
951 12
零基础保姆级教程!手把手教你免费玩转Linux CentOS安装+学习环境搭建(附避坑指南)
|
10月前
|
存储 网络协议 Linux
【Linux】进程IO|系统调用|open|write|文件描述符fd|封装|理解一切皆文件
本文详细介绍了Linux中的进程IO与系统调用,包括 `open`、`write`、`read`和 `close`函数及其用法,解释了文件描述符(fd)的概念,并深入探讨了Linux中的“一切皆文件”思想。这种设计极大地简化了系统编程,使得处理不同类型的IO设备变得更加一致和简单。通过本文的学习,您应该能够更好地理解和应用Linux中的进程IO操作,提高系统编程的效率和能力。
457 34

热门文章

最新文章