Linux(fork+exec创建进程)

1.进程创建

内核设计与实现43页;

执行了3次ps -f ,ps -f的父进程的ID(PPID)都是一样的,即bash.

实际上Linux上这个bash就是不断的复制自身,然后把复制出来的用exec替换成想要执行的程序(比如ps);

运行ps,发现ps是bash的一个子进程;原因就是bash把自己复制一份,然后替换成ps;

替换,这里就体现了写时拷贝的意义,如果全部都要替换,那么最开始的复制是没有意义的;
注意,用了写时拷贝就只复制了几个页表的映射,内容还没有复制,然后执行了替换exec.

在Linux新的进程的产生过程(进程创建):

fork:复制进程
exec系列:将当前进程替换为另外一个进程.

2.进程替换exec系列介绍

ececl,execlp,execle,execv,execvp   //库函数

execve   //系统调用

例1:execl

which  ps

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main()
{
   printf("main pid=%d\n",getpid());
   execl("/usr/bin/ps","ps","-f",(char *)0);
   exit(0);
}

注意,就是原来的程序换成了ps程序,但是PCB没有改变,但是PCB里面的有些值被修改了,比如pcb中程序的名字换成了新进程的名字;

思考1,新的进程从哪里执行呢?

新的进程从主函数的第一行开始执行,也就是ps程序的主函数的第一行代码开始执行,这个和fork方法不一样,fork返回以后,从fork所在位置开始执行;
所以可以直接在execl下面打印一个失败,如果成功就根本不会执行到这里;
printf("execl error\n");

思考2:如果将execl里面的第二个参数改为"abc",程序还能否执行?

可以正常执行,只是程序命令改为了abc;

execl("/usr/bin/ps","abc","-f",(char *)0);

思考3:如果将execl里面的第一个参数改为"abc",程序还能否执行?

只要execl第一个参数不出错,第一个参数如果出错了,你就找不到这个程序了;  那么就运行不成功了;比如:

execl("/usr/bin/abc","ps","-f",(char *)0);

一定要在exec替换函数之后写:

printf("exec  error!\n");

失败了才运行到这一句.

例2:execlp

execlp("ps","ps","-f",(char *)0);

只给文件名,不需要给文件路径,因为它可以去环境变量PATH所指的位置去搜索;

注意,第一句,虽然有两个ps,但是不能省,第一个代表我们启动的是ps(去环境变量下搜索),第二个代表的是替换的程序也就是新程序的名字;

例3:execle

和execl一样的参数,只是多了最后一个环境变量.

//int main(int argc,char *argv[],char *envp[])
execle("/usr/bin/ps","ps","-f",(char *)0,envp);

思考:如何使用自己的环境变量呢?

例4:execv

//int main(int argc,char *argv[],char *envp[])

char *myargv[]={"ps","-f",0};
execv("/usr/bin/ps",myargv);

例5:execvp

第一个参数只要文件名,不要路径;

//int main(int argc,char *argv[],char *envp[])

char *myargv[]={"ps","-f",0};
execvp("ps",myargv);

看帮助手册,execvpe是GNU的扩展,不通用,所以我们这里不做介绍.

例6:execve

int execve(const char * path, char* const argv[],char* const envp[]); //系统调用

代码如下:

//int main(int argc,char *argv[],char *envp[])

char *myargv[]={"ps","-f",0};
execve("/usr/bin/ps",myargv,envp);

3.总结:

//pathname:新替换的程序的路径+名字
//arg :传给新程序主函数的第一个参数,一般为程序的名字
//arg 后面是剩余参数列表,参数个数可变,必须以空指针作为最后一个参数
int execl(const char* pathname, const char * arg,...);
int execlp(const char* file, const char * arg,...);
int execle(const char* pathname, const char * arg,...,char* const envp[]);
int execv(const char * pathname, char* const argv[]);
int execvp(const char * file, char* const argv[]);
int execve(const char * pathname, char* const argv[],char* const envp[]); //系统调用

//前五个是库函数,最后一个是系统调用,所以本质上上面5个都是通过第六个系统调用实现的

也就是说,上面5个都是调用的execve,不过都是把参数放进数组,然后把数组传递给这个系统调用execve;
也就是说,这些方法没有本质区别;
也就是说,本质上只有一个替换方法,就是execve;

注意,写一个(char *)NULL也是可以的;

助记:
l(list)     参数地址列表,以空指针结尾
v(vector)   存放各个参数地址的指针数组的地址
p(path)   按PATH环境变量指定的目录搜索可执行文件
e(enviroment)  存放环境变量字符地址的指针数组的地址

4.环境变量也可以自己添加

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <assert.h>
#include <assert.h>
int main()
{
   //存放传给新程序主函数的参数
   char * myargv[]={"ps","-f",(char *)0};
   //存放传给新程序主函数的环境变量
   char *myenvp[]={"MYSTR=hello","VAL=100",(char *)0};
   printf("main pid=%d\n",getpid());
   //excel执行成功不返回,失败返回错误码
   execve("/bin/ps",myargv,myenvp);
   perror("execve error\n");
   exit(0);
}

5.进程替换的应用

写一个程序main.c,运行起来之后替换执行test程序(test打印参数内容)

//test.c
 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 
 
int main(int argc,char *argv[])
 {
 printf("test start and test_pid=%d\n",getpid());

 int i=0;
 for(;i<argc;i++)
 {
 printf("argv[%d]=%s\n",i,argv[i]);
 }

 printf("test end\n");
 exit(0);
 }

//main.c
#include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>

 int main(int argc,char *argv[],char *envp[])
 {
    printf("main start and mainpid=%d\n",getpid());

    execl("./test","./test","a","b","c",(char *)0);
    perror("execl error");执行结果如下: 

    printf("main end\n");//执行了之后替换了程序,这一句不会被执行到;
    exit(0);
 }

6.进程创建示例

1).创建ps命令-execl的使用(结合fork 1)

exec系列单独是能使用的,但是没有多大意义.通常我们会结合fork一起使用;

fork+exec()是Linux上创建新进程的方式;

 #include<stdio.h>
 #include<stdlib.h>
 #include<unistd.h>
 #include<assert.h>
 #include<wait.h>

int main()
{
    printf("main pid=%d,ppid=%d\n",getpid(),getppid());
    pid_t pid=fork();
    assert(pid!=-1);

   if(pid==0)
   {
     printf("child pid=%d,ppid=%d\n",getpid(),getppid());
    // execl("/bin/ps","-f",(char *)0);//省略了ps也对,但是最好写成上面的;
    execl("/usr/bin/ps","ps","-f",NULL);
    printf("execl error");
    exit(0);
   }

   wait(NULL);

   exit(0);
 }

比如我们在子进程退出前:

    printf("child end!\n");

我们发现执行不到这一句,因为去执行ps去了,然后从ps退出进程了,除非execl执行失败.

2.fork和exec联合使用创建一个全新的进程(结合  fork2 )

当前主程序main通过fork复制产生一个子进程,子进程用新程序"newmain"替换自身;  (newmain:打印参数内容和环境变量)

//main.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <assert.h>
#include <wait.h>
int main(int argc,char *argv[],char *envp[])
{
    printf("main pid=%d\n",getpid());
    pid_t pid=fork();
    assert(pid!=-1);
    if(pid==0)
   {
         char *myargv[]={"newmain","hello","abc","123",(char *)0};
         //char *myenvp[]={"MYSTR=hello","VAL=100",(char *)0};
         execve("./newmain",myargv,envp);
         perror("execl error");
         exit(0);
   } 
    wait(NULL);
    printf("main over\n");
    exit(0);

//newmain.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc,char *argv[],char *envp[])
{
    printf("newmain pid=%d\n",getpid());
    int i=0;
    printf("argc=%d\n",argc);
    
    for(;i<argc;i++)
    {
      printf("argv[%d]=%s\n",i,argv[i]);
    }
    for(i=0;envp[i]!=NULL;i++)
   {
        printf("envp[%d]=%s\n",i,envp[i]);
   } 
    exit(0);
}

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

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

相关文章

竞赛选题 题目:基于深度学习的中文对话问答机器人

文章目录 0 简介1 项目架构2 项目的主要过程2.1 数据清洗、预处理2.2 分桶2.3 训练 3 项目的整体结构4 重要的API4.1 LSTM cells部分&#xff1a;4.2 损失函数&#xff1a;4.3 搭建seq2seq框架&#xff1a;4.4 测试部分&#xff1a;4.5 评价NLP测试效果&#xff1a;4.6 梯度截断…

dbeaver连接amabri-hbase

目录 尝试过程 解决之道 总结 尝试过程 注意此章节为记录试错过程&#xff0c;无需跟随操作&#xff0c;仅作试错记录。真正操作方法请看“解决之道”章节 环境ambari安装的hbase2.1.6 使用apche phoenix默认驱动配置 备注&#xff1a;Apache Phoenix 是一个开源的、基于…

fastadmin 如何引入自己的js

在需要的界面中&#xff1a;如何实例说明&#xff1a; 中<script> function zhuruJs(url) { let temp document.createElement( script ); temp.setAttribute( type, text/javascript" );temp.src urL; document.head . appendChild(temp); zhuruJs(location…

ffmpeg 免安装,配置环境变量

1、下载ffmpeg https://download.csdn.net/download/qq284489030/88579595 2、解压 解压ffmpeg-4.4-essentials_build.zip到目标文件夹&#xff0c;比如 d:\apps下&#xff1b; 3、配置环境变量 &#xff08;1&#xff09;电脑桌面鼠标右键点击“此电脑”&#xff0c;弹出…

脚本格式问题记录

服务器上的一些脚本迁移到其他服务上发生的小问题 问题&#xff1a;执行一个在win10系统编写好的shell脚本&#xff0c;放到Linux上执行报错如下&#xff1a; bash: ./xxx.sh: /bin/bash^M: bad interpreter: No such file or directory 原因&#xff1a;window系统写的脚本&a…

iMazing是什么软件?2024最新版本如何下载

iMazing是一款功能强大的iOS设备管理软件&#xff0c;它可以帮助用户备份和管理他们的iPhone、iPad或iPod Touch上的数据。除此之外&#xff0c;它还可以将备份数据转移到新的设备中、管理应用程序、导入和导出媒体文件等。本文将详细介绍iMazing的功能和安全性&#xff0c;并教…

实验一 SAS 基本操作和数据表的导入 2023-11-29

一、上机目的 熟悉SAS的集成环境并掌握它的基本操作。理解SAS程序的结构&#xff0c;理解其中的过程&#xff0c;过程选项&#xff0c;语句&#xff0c;语句选项等概念&#xff0c;掌握SAS编程技术。 二、上机内容 主要有SAS操作界面、SAS窗口操作、SAS菜单操作、SAS按钮操作…

用函数初始化数组

将数组全部初始化为相同值 对于一般情况 一般是用函数&#xff0c;传什么数就初始化为什么数 #define _CRT_SECURE_NO_WARNINGS 1 #include<stdio.h> void init(int arr[], int len, int num) {int i;for (i 0; i < len; i){arr[i] num;} } int main() {int arr[…

如何判断哪种屋顶适合安装光伏板?

随着国家对可再生能源的推广和大力发展&#xff0c;光伏板开始被越来越多人所熟知。而将光伏板安装在家庭楼顶上&#xff0c;不仅可以有效节省土地和楼房面积&#xff0c;还能够为家庭提供更多的经济和环保效益&#xff0c;成为了越来越多人的选择。哪种屋顶适合安装光伏板呢&a…

php爬虫实现把目标页面变成自己的网站页面

最近又被烦的不行&#xff0c;琐事不断&#xff0c;要是比起懒来一个人比一个人懒&#xff0c;但是懒要转换成动力啊&#xff0c;能让自己真正的偷懒&#xff0c;而不是浪费时间。每天还是需要不断的学习的&#xff0c;才能更好的提高效率&#xff0c;把之前做的简单小功能爬虫…

第二部分 系统管理篇

文件和目录管理 Linux基础 在Linux操作系统中&#xff0c;一切都是文件。Linux文件是区分大小写的。 Linux文件的拓展名和它的种类没有任何关系 Linux的目录结构为树状结构&#xff0c;顶级的目录为根目录“/”。 文件类型 用file命令查看文件类型 文件操作命令 1.mkdir创…

el-select实现分屏效果

动态绑定class值 &#xff0c;多种判断 :class"type 8 ? home-stye-2 : type 24 ? home-stye-1 : home-stye-3" <div class"home-right-top"><div class"home-right-top-video"><el-row :gutter"20"><el-c…

Android进阶之路 - TextView文本渐变

那天做需求的时候&#xff0c;遇到一个小功能&#xff0c;建立在前人栽树&#xff0c;后人乘凉的情况下&#xff0c;仅用片刻就写完了&#xff1b;说来惭愧&#xff0c;我以前并未写过文本渐变的需求&#xff0c;脑中也仅有一个shape渐变带来的大概思路&#xff0c;回头来看想着…

福州大学《嵌入式系统综合设计》 实验九:ROI视频编码

一、实验目的 ROI视频编码即感兴趣区域视频编码&#xff0c;即针对感兴趣区域进行重点编码&#xff0c;提高编码质量&#xff0c;而对非感兴趣区域采用低质量编码。通过这种方法可以降低码率。本实验即让同学们能够在算能的FFMPEG接口下实现基于ROI的视频编码。 二、实验内容…

离散化笔记

文章目录 离散化的适用条件离散化的意思AcWing 802. 区间和CODECODE2 离散化的适用条件 离散化用于区间求和问题对于数域极大&#xff0c;而数的量很少的情况下 离散化的意思 背景&#xff1a;对于一个极大数域上的零星几个数进行操作后&#xff0c;求某段区间内的和 其实意思…

JSch线上出现com.jcraft.jsch.JSchException: channel is not opened.问题分析

JSch线上出现com.jcraft.jsch.JSchException: channel is not opened.问题分析 文章目录 JSch线上出现com.jcraft.jsch.JSchException: channel is not opened.问题分析1. 背景1.系统使用jsch这个框架做文件发送以及远程命令执行的操作,系统一直运行正常,直到某一个环境发现 2.…

关于我司在上海物联网行业协会展厅展示项目案例

1 项目背景 上海市物联网行业协会&#xff08;SIOT&#xff09;是由本市物联网行业同业企业及其他相关经济组织自愿组成、实行行业服务和自律管理的非营利性社会团体法人&#xff0c;于2012年&#xff0c;经上海市经济和信息化委同意&#xff0c;在上海市社团局登记成立。 本…

【精选】Spring整合MyBatis,Junit 及Spring 事务Spring AOP面向切面详解

Spring整合MyBatis 搭建环境 我们知道使用MyBatis时需要写大量创建SqlSessionFactoryBuilder、SqlSessionFactory、SqlSession等对象的代码&#xff0c;而Spring的作用是帮助我们创建和管理对象&#xff0c;所以我们可以使用Spring整合MyBatis&#xff0c;简化MyBatis开发。 …

SDK emulator directory is missing

要进行uniapp真机测试&#xff0c;不得不安装配置一下安卓开发环境 &#xff0c;搞一个模拟器。。。然后又是各种坑。。对比来对比去还是IOS的环境使用着舒服&#xff0c;XCODE下载好&#xff0c;一切重点就是在编码了。。 安卓这个脑残货呀&#xff0c;哎&#xff0c;各种安装…

数据挖掘之时间序列分析

一、 概念 时间序列&#xff08;Time Series&#xff09; 时间序列是指同一统计指标的数值按其发生的时间先后顺序排列而成的数列&#xff08;是均匀时间间隔上的观测值序列&#xff09;。 时间序列分析的主要目的是根据已有的历史数据对未来进行预测。 时间序列分析主要包…