Linux(fork+exec创建进程)

news2025/1/19 8:01:07

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.coloradmin.cn/o/1269436.html

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

相关文章

深度学习-模型调试经验总结

1、 这句话的意思是&#xff1a;期望张量的后端处理是在cpu上&#xff0c;但是实际是在cuda上。排查代码发现&#xff0c;数据还在cpu上&#xff0c;但是模型已经转到cuda上&#xff0c;所以可以通过把数据转到cuda上解决。 解决代码&#xff1a; tensor.to("cuda")…

vuepress-----7、发布在GitHub

# 7、发布在GitHub 在你的项目中&#xff0c;创建一个如下的 deploy.sh 文件&#xff08;请自行判断去掉高亮行的注释&#xff09;: #!/usr/bin/env sh# 确保脚本抛出遇到的错误 set -e# 生成静态文件 npm run docs:build# 进入生成的文件夹 cd docs/.vuepress/dist# 如果是发…

attention中Q,K,V的理解

第一种 1.首先定义三个线性变换矩阵&#xff0c;query&#xff0c;key&#xff0c;value&#xff1a; class BertSelfAttention(nn.Module):self.query nn.Linear(config.hidden_size, self.all_head_size) # 输入768&#xff0c; 输出768self.key nn.Linear(config.hidde…

python实验3 石头剪刀布游戏

实验3&#xff1a;石头剪刀布游戏 一、实验目的二、知识要点图三、实验1. 石头剪刀布2. 实现大侠个人信息 一、实验目的 了解3类基本组合数据类型。理解列表概念并掌握Python中列表的使用。理解字典概念并掌握Python中字典的使用。运用jieba库进行中文分词并进行文本词频统计。…

COGVLM论文解读(COGVLM:VISUAL EXPERT FOR LARGE LANGUAGE MODELS)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、摘要二、引言三、模型方法1、模型思路2、融合公式 四、训练方法总结 前言 2023年5月18日清华&智谱AI发布并开源VisualGLM-6B以来&#xff0c;清华KEG&…

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

文章目录 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 梯度截断…

BUUCTF [GXYCTF2019]BabyUpload 1详解(.htaccess配置文件特性)

题目环境&#xff1a;查看题目源码 SetHandler application/x-httpd-php 通过源码可以看出这道文件上传题目主要还是考察.htaccess配置文件的特性 倘若不先上传.htaccess配置文件&#xff0c;那么后台服务器就无法解析php代码 这个是需要注意的 .htaccess配置文件特性 概述来说…

函数学习 PTA 1使用函数输出一个整数的逆序数;3判断满足条件的三位数;5使用函数求余弦函数的近似值

其实一共有五道题&#xff0c;但那两道实在太过简单&#xff0c;也不好意思打出来给大家看&#xff0c;那么这篇博客&#xff0c;就让我一次性写三道题吧&#xff01;也当是个小总结&#xff0c;睡前深思。 6-1 使用函数输出一个整数的逆序数 本题要求实现一个求整数的逆序数的…

STM32F407-14.3.6-01输入捕获模式

输入捕获模式 在输入捕获模式下&#xff0c;当相应的 ICx⑦ 信号检测到跳变沿后&#xff0c;将使用捕获/比较寄存器 (TIMx_CCRx⑪) 来锁存计数器的值。发生捕获事件时&#xff0c;会将相应的 CCXIF⑬ 标志&#xff08;TIMx_SR 寄存器&#xff09;置 1&#xff0c; 并可发送中断…

云时空社会化商业 ERP 系统 Shiro 反序列化漏洞复现

0x01 产品简介 时空云社会化商业ERP&#xff08;简称时空云ERP&#xff09; &#xff0c;该产品采用JAVA语言和Oracle数据库&#xff0c; 融合用友软件的先进管理理念&#xff0c;汇集各医药企业特色管理需求&#xff0c;通过规范各个流通环节从而提高企业竞争力、降低人员成本…

SpringMVC多种类型数据响应

SpringMVC多种类型数据响应入门 1.概念 RequestMapping 作用&#xff1a;用于建立请求URL和处理请求方法之间的对应关系 位置&#xff1a; 类上&#xff0c;请求URL的第一级访问目录。此处不写的话&#xff0c;就相当于应用的根目录 方法上&#xff0c;请求URL的第二级访问目…

【二叉树】Leetcode 199. 二叉树的右视图

力扣题目链接 解题思路 一开始&#xff0c;我以为只需要依次遍历最右边一列所有数即可&#xff0c;写出来的代码也通过了样例&#xff1a; class Solution { public:vector<int> rightSideView(TreeNode* root) {vector<int> ans;TreeNode* temp root;while(tem…

游戏缺少d3dx9_43.dll修复方法分享,快速解决dll缺失问题

在计算机使用过程中&#xff0c;我们常常会遇到一些错误提示&#xff0c;其中之一就是“找不到d3dx9_43.dll文件”。这个错误通常出现在运行某些游戏或应用程序时&#xff0c;d3dx9_43.dll是一个动态链接库文件&#xff0c;它是DirectX 9的一部分&#xff0c;用于支持游戏中的3…

LeetCode 7 整数反转

题目描述 整数反转 给你一个 32 位的有符号整数 x &#xff0c;返回将 x 中的数字部分反转后的结果。 如果反转后整数超过 32 位的有符号整数的范围 [−231, 231 − 1] &#xff0c;就返回 0。 假设环境不允许存储 64 位整数&#xff08;有符号或无符号&#xff09;。 示例…

U-boot(七):U-boot移植

本文主要探讨基于210官方U-boot源码移植。 移植基础 tar -jxvf android_uboot_smdkv210.tar.bz2cd u-boot-samsung-devrm -rf onenand_ipl onenand_bl1 lib_avr32 lib_blackfin lib_i386 lib_m68k lib_mips lib_microblaze lib_nios lib_nios2 lib_ppc lib_sh lib_sparccd bo…

Ubuntu防止休眠和挂起(笔记)

目录 1 动机2 禁用休眠3 解除休眠 1 动机 我要将 饿啊人制作成 noah-mp 的区域运行强迫&#xff0c;但是跑的慢&#xff0c;一晚上两天。后来发现是因为电脑自动 supend 了。Ubuntu 在电源那里最多只能设置 2 小时的防止休眠&#xff0c;2小时候又自动休眠&#xff0c;严重影响…

mongodb基本操作命令

mongodb快速搭建及使用 1.mongodb安装1.1 docker安装启动mongodb 2.mongo shell常用命令2.1 插入文档2.1.1 插入单个文档2.1.2 插入多个文档2.1.3 用脚本批量插入 2.2 查询文档 前言&#xff1a;本篇默认你是对nongodb的基础概念有了了解&#xff0c;操作是非常基础的。但是与关…

Opencv手势控制音量!附源码!

效果演示&#xff1a; 废话不多说&#xff01;直接上源码&#xff01;下面写有所有代码注释&#xff01;&#xff01; import cv2 import mediapipe as mp #它包含了各种预训练的机器学习模型&#xff0c;可以用于姿势估计、手势识别等任务 from ctypes import cast, POINTE…

P8649 [蓝桥杯 2017 省 B] k 倍区间(前缀和+优化(桶分类))

分析&#xff1a; &#xff08;1&#xff09;任意连续子序列可用两个前缀和的差来表示 &#xff08;2&#xff09;判断该子序列是否为k的倍数 p1-p2 模 0 (mod k) 等价于&#xff1a;前缀和模 k 是否同余 &#xff08;3&#xff09;同余的任意两前缀和组合的序列均满足…

[React] 2023年最新面试题

[React] 2023年最新面试题 1. class 组件与函数组件的区别2. react 18 新特性有那些?新增 createRoot API自动批处理过渡更新新的Hook 3. redux 和 react-redux 的区别4. redux 中间件的原理5. setState 发生了什么 &#xff0c;render 函数做了什么6. 虚拟DOM&#xff0c; Fi…