OS | 第5章 插叙:进程API

news2024/12/23 0:13:15

OS | 第5章 插叙:进程API

文章目录

  • OS | 第5章 插叙:进程API
    • 5.1 fork()系统调用
      • 代码过程分析
    • 5.2 wait()系统调用
      • 5.3 exec() 系统调用
      • 执行过程
    • 为什么这样设计API?
      • shell执行过程
      • shell 重定向
      • pipe()
      • >>>>> 欢迎关注公众号【三戒纪元】 <<<<<

UNXI 系统采用了一种非常有趣的创建新进程的方式,通过一对系统调用: fork()exec(),还可以通过第3个系统调用 wait(),来等待其创建的子进程执行完成。

5.1 fork()系统调用

系统调用 fork()用于创建新进程。

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

int main(int argc, char *argv[]) {
  printf("hello Randy (pid:%d) \n", (int)getpid());
  int rc = fork();
  if (rc < 0)  // fork failed; exit
  {
    fprintf(stderr, "fork failed!\n");
    exit(1);
  } else if (rc == 0) {  // child (new process)
    printf("hello, I am child (pid:%d)\n", (int)getpid());
  } else {  // parent goes down this path (main)
    printf("hello, I am parent of %d (pid:%d)\n", rc, (int)getpid());
  }
  return 0;
}

运行结果:

hello Randy (pid:12242) 
hello, I am parent of 12243 (pid:12242)
hello, I am child (pid:12243)

代码过程分析

  1. 程序开始运行时, 进程输出一条 “hello Randy”信息,以及自己的进程描述符(process identifier, PID),该进程PID为12242;
  2. 进程调用了fork()系统调用,创建了1个新的线程。新创建的线程与调用线程几乎完全一样,对操作系统来说,有2个完全一样的程序在运行,并从fork()系统调用中返回;
  3. 新创建的进程称为子进程(child),原来的线程为父进程(parent)。子进程直接从fork()系统调用返回,好像自己调用了fork();
  4. 父进程和子进程接着分别运行剩余代码。

子进程并不是完全拷贝父进程:虽然子进程有自己的地址空间(拥有自己的私有内存)、寄存器、程序计数器等,但是它从fork()返回的值是不同的。

父进程获得的返回值是子进程的PID,子进程获得的返回值是0。

CPU调度程序(scheduler)决定了某个时刻哪个进程被执行,所以父进程和子进程谁先打印不确定。

5.2 wait()系统调用

有时候父进程需要等待子进程执行完毕,由wait()系统调用完成

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>

int main(int argc, char *argv[]) {
  printf("hello Randy (pid:%d) \n", (int)getpid());
  int rc = fork();
  if (rc < 0)  // fork failed; exit
  {
    fprintf(stderr, "fork failed!\n");
    exit(1);
  } else if (rc == 0) {  // child (new process)
    printf("hello, I am child (pid:%d)\n", (int)getpid());
  } else {  // parent goes down this path (main)
    int wc = wait(NULL);
    printf("hello, I am parent of %d (wc:%d) (pid:%d)\n", rc, wc,  (int)getpid());
  }
  return 0;
}

执行结果:

hello Randy (pid:14580) 
hello, I am child (pid:14581)
hello, I am parent of 14581 (wc:14581) (pid:14580)

父进程调用wait(),延迟自己的执行,直到子进程执行完毕,wait()才返回父进程。

5.3 exec() 系统调用

exec() 有几种变体:execl()、execle()、execlp()、execv()、execvp()。

这个系统调用可以让子进程执行与父进程不同的程序。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>

#include <string.h>

int main(int argc, char *argv[]) {
  printf("hello Randy (pid:%d) \n", (int)getpid());
  int rc = fork();
  if (rc < 0)  // fork failed; exit
  {
    fprintf(stderr, "fork failed!\n");
    exit(1);
  } else if (rc == 0) {  // child (new process)
    printf("hello, I am child (pid:%d)\n", (int)getpid());
    char *myargs[3];
    myargs[0] = strdup("wc");        // program: "wc"(word count)
    myargs[1] = strdup("fork.cpp");  // program: "wc"(word count)
    myargs[2] = NULL;                // program: "wc"(word count)
    execvp(myargs[0], myargs);       // runs word count
    printf("this shouldn't print out");
  } else {                     // parent goes down this path (main)
    int wc = wait(NULL);
    printf("hello, I am parent of %d (wc:%d) (pid:%d)\n", rc, wc,  (int)getpid());
  }
  return 0;
}

执行结果:

./randy 
hello Randy (pid:15334) 
hello, I am child (pid:15335)
 27 117 916 fork.cpp
hello, I am parent of 15335 (wc:15335) (pid:15334)

子进程调用execvp()运行字符计数程序wc。

执行过程

  1. exec() 给定可执行程序的名称(如wc)及需要的参数后,会从可执行程序中加载代码和静态数据,并用它覆写自己的代码段(以及静态数据),堆、栈及其他内存空间也会被重新初始化。

  2. 操作系统执行该程序,将参数通过argv 传递给该进程。

它并没有创建新进程,而是直接将当前运行的程序替换为不同的运行程序(wc)。

子进程执行exec()之后,几乎就像源程序从未运行过一样,对exec() 的成功调用永远不会返回。

为什么这样设计API?

这种分离的fork()和exec()的做法在构建UNIX shell 的时候非常有用,这给了shell 在fork之后 exec 之前运行代码的机会,这些代码可以在运行新程序之前改变环境,从而让其他功能易于实现。

比如:

wc fork.cpp > count_fork_cpp.txt

shell执行过程

shell 是一个用户程序(包括 tcsh、bash、zsh等)

  1. 首先显示一个提示符(prompt)
  2. 等待用户输入。输入命令或参数
  3. shell在文件系统中找到这个可执行程序,调用fork()创建新进程;
  4. 调用exec()的某个变体执行这个可执行程序
  5. 调用wait()等待该命令完成

shell从wait()返回并再次输出一个提示符,等待用户输入下一条命令。

shell 重定向

上面的例子,wc的结果被重定向到count_fork_cpp.txt中。

重定向的工作原理,是基于操作系统管理文件描述符方式的假设。

UNIX系统从0开始寻找可以使用的文件描述符。

下面的例子中STDOUT_FILENO将成为第一个可用的文件描述符,因此在open()被调用时,得到赋值。然后子进程向标准输出文件描述符的写入(如printf()),都会被透明地转向新打开的文件,若不是屏幕。

#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>

int main(int argc, char *argv[]) {
  int rc = fork();
  if (rc < 0)  // fork failed; exit
  {
    fprintf(stderr, "fork failed!\n");
    exit(1);
  } else if (rc == 0) {  // child : redict standard  output to a file
    close(STDOUT_FILENO);
    open("./redirect.output", O_CREAT | O_WRONLY | O_TRUNC, S_IRWXU);
    
    // now exec "wc"
    printf("hello, I am child (pid:%d)\n", (int)getpid());
    char *myargs[3];
    myargs[0] = strdup("wc");        // program: "wc"(word count)
    myargs[1] = strdup("fork.cpp");  // program: "wc"(word count)
    myargs[2] = NULL;                // program: "wc"(word count)
    execvp(myargs[0], myargs);       // runs word count
    printf("this shouldn't print out");
  } else {                     // parent goes down this path (main)
    // printf("hello, I am parent of %d (pid:%d)\n", rc, (int)getpid());
    int wc = wait(NULL);
    printf("hello, I am parent of %d (wc:%d) (pid:%d)\n", rc, wc,  (int)getpid());
  }
  return 0;
}

执行结果:

Randy@Jeff:~$ ./randy 
hello, I am parent of 17118 (wc:17118) (pid:17117)
Randy@Jeff:~$ cat redirect.output 
  46  227 1592 fork.cpp

pipe()

UNIX管道也是用类似的方式实现的,用的是pipe()系统调用。

一个进程的输出被连接到一个内核管道(pipe)上(队列),另一个进程的输入也被连接到了同一个管道上。

因此前一个进程的输出无缝地作为后一个进程的输入,许多命令可以用这种方式串联在一个,共同完成某项任务。如从一个文件中查找某个词,并统计其出现的次数:

grep -o randy file | wc -l

>>>>> 欢迎关注公众号【三戒纪元】 <<<<<

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

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

相关文章

YOLOv5:解读metrics.py

YOLOv5&#xff1a;解读metrics.py 前言前提条件相关介绍metrics.pyfitnesssmoothbox_iouConfusionMatrix ★ ★ \bigstar\bigstar ★★bbox_iou ★ ★ \bigstar\bigstar ★★compute_apap_per_class&#xff08;难度&#xff1a; ⋆ ⋆ ⋆ ⋆ ⋆ \star\star\star\star\star ⋆…

openpnp - 底部相机高级矫正后,底部相机看不清吸嘴的解决方法

文章目录 openpnp - 底部相机高级矫正后,底部相机看不清吸嘴的解决方法概述解决思路备注补充 - 新问题 - N1吸嘴到底部相机十字中心的位置差了很多END openpnp - 底部相机高级矫正后,底部相机看不清吸嘴的解决方法 概述 自从用openpnp后, 无论版本(dev/test), 都发现一个大概…

mac建议装双系统吗,详细分析苹果电脑双系统的利弊

mac建议装双系统吗&#xff0c;Mac电脑上安装双系统有哪些利弊呢&#xff0c;一起来看看吧&#xff01; 苹果Mac电脑安装双系统利&#xff1a; 1、用来办公更加方便&#xff1a;苹果系统功能也是很强大的&#xff0c;但是用来办公非常不方便&#xff0c;是由于一些常用的exe软…

Error: The project seems to require yarn but it‘s not installed.

把之前做过的vue项目拷贝到新电脑上&#xff0c;运行启动命令后发现报了如下错误&#xff1a; 我是这么解决的&#xff1a; 是因为项目中存在yarn.lock 文件&#xff0c;先把这个文件删除掉。 把这个文件删除后&#xff0c;执行如下命令&#xff1a; npm install -g yarn 下…

Docker Desktop 设置镜像环境变量

点击run 展开Optional settings container name &#xff1a;容器名称 Ports&#xff1a;根据你需要的端口进行输入&#xff0c;不输入则默认 后面这个 比如我这个 5432 Volumes&#xff1a;卷&#xff0c;也就是做持久化 需要docker 数据保存的地方 Environment variables…

TCP的滑动窗口协议有什么用?

分析&回答 滑动窗口协议&#xff1a; TCP协议的使用维持发送方/接收方缓冲区 缓冲区是 用来解决网络之间数据不可靠的问题&#xff0c;例如丢包&#xff0c;重复包&#xff0c;出错&#xff0c;乱序 在TCP协议中&#xff0c;发送方和接受方通过各自维护自己的缓冲区。通…

react16之前diff算法的理解和总结

此篇文章所讨论的是 React 16 以前的 Diff 算法。而 React 16 启用了全新的架构 Fiber&#xff0c;相应的 Diff 算法也有所改变&#xff0c;本片不详细讨论Fiber。 fiber架构是为了支持react进行可中断渲染&#xff0c;降低卡顿&#xff0c;提升流畅度。 react16之前的版本&…

什么是standard cell (标准单元) ?

参考文章&#xff1a; 聊一聊芯片后端的标准单元-standard cell - 知乎 (zhihu.com) standard cell中的7T和9T中的"T"指的是什么&#xff1f;或者是什么的缩写&#xff1f; - Layout讨论区 - EETOP 创芯网论坛 (原名&#xff1a;电子顶级开发网) - 数字后端基本概念介…

qemu-system-x86_64 命令创建虚拟机,报gtk initialization failed的

因为是ssh命令行启动&#xff0c;增加--nographic # /opt/debug/bin/qemu-system-aarch64 -machine virt-6.2 -qmp tcp:localhost:1238,server,nowait --nographic configure accelerator virt-6.2 start machine init start cpu init start add rom file: virtio-net-pci…

GIT实战篇,教你如何使用GIT可视化工具

系列文章目录 手把手教你安装Git&#xff0c;萌新迈向专业的必备一步 GIT命令只会抄却不理解&#xff1f;看完原理才能事半功倍&#xff01; 快速上手GIT命令&#xff0c;现学也能登堂入室 GIT实战篇&#xff0c;教你如何使用GIT可视化工具 系列文章目录一、GIT有哪些常用工具…

漆包线工厂云MES解决方案

漆包线行业老板痛点&#xff1a; 1.漆包线比较传统的行业&#xff0c;一般都是靠人工去管理&#xff0c;老板想及时知道工厂的生产&#xff0c;销售、出入库、库存情况&#xff1b; 2.型号多称重打印易错&#xff0c;没有系统前 &#xff1a;称重打印&#xff0c;出入库&#x…

解决github连接不上的问题

改 hosts 我们在浏览器输入 GitHub 的网址时&#xff0c;会向 DNS 服务器发送一个请求&#xff0c;获取到 GitHub 网站所在的服务器 IP 地址&#xff0c;从而进行访问。 就像你是一名快递员&#xff0c;在送快递前要先找中间人询问收件人的地址。而 DNS 就是这个告诉你目标地址…

java开发之个微机器人的二次开发

简要描述&#xff1a; 取消消息接收 请求URL&#xff1a; http://域名地址/cancelHttpCallbackUrl 请求方式&#xff1a; POST 请求头Headers&#xff1a; Content-Type&#xff1a;application/json 参数&#xff1a; 参数名类型说明codestring1000成功&#xff0c;1…

顶尖211“小清华”!强过985,不要错过它!

一、学校及专业介绍 西安电子科技大学&#xff08;Xidian University&#xff09;&#xff0c;简称“西电” &#xff0c;位于陕西省西安市&#xff0c;是中央部属高校&#xff0c;直属于教育部&#xff0c;为全国重点大学&#xff0c;位列国家“双一流”“211工程”&#xff…

宝塔面板开心版出问题升级到正版的解决方案,有效解决TypeError: ‘NoneType‘ object is not subscriptable

服务器之前图开心装了个宝塔面板的开心版&#xff0c;前几天突然出现问题&#xff0c;报错TypeError: ‘NoneType’ object is not subscriptable。没法正常打开软件商店&#xff0c;也没法修复和升级系统&#xff0c;很烦躁。因为里面很多业务还在跑&#xff0c;实在不想重装。…

呜呜呼呼无无话

姓名和手机号脱敏 function nameDesen(value) {if (!value) return return value.substring(0, 1) new Array(value.length).join(*) } const bklnameDesen(宝矿力) console.log(bkl) //宝**function telephoneDesen(value) {if (!value) return value value.toString()ret…

伦敦金投资中挂单的优势

在伦敦金交易中&#xff0c;其实挂单具有无可比拟的优势&#xff1f;首先&#xff0c;如果我们不是全职的投资者&#xff0c;我们在交易时间内可能有其他的工作&#xff0c;那么挂单就可以帮助我们捕捉到市场的机会&#xff0c;即便我们不一定会坐在电脑面前&#xff0c;也可以…

Vue中的指令

指令 指令 (Directives) 是带有 v- 前缀的特殊 attribute。指令 attribute 的值预期是单个 JavaScript 表达式。指令的职责是&#xff0c;当表达式的值改变时&#xff0c;将其产生的连带影响&#xff0c;响应式地作用于 DOM。 常用指令预期简短介绍v-showany显示隐藏元素&…

旅游攻略APP外包开发功能

旅游攻略APP是帮助旅行者计划和享受旅行的工具&#xff0c;下面列出了一些常见的旅游攻略APP功能&#xff0c;以及在上线这类应用时需要注意的问题&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合作。 常见…

6517B|吉时利keithley 6517B高阻表

181/2461/8938Keithley 6517B 静电计 / 高阻表是全球科研实验室灵敏测量的标准。吉时利在弱电测量方面拥有 60 多年的专业经验&#xff0c;吉时利静电计能够可靠地测量最低 10 aA (1010-18A) 电流、最低 1 fC 电荷&#xff0c;并支持最高 1018Ω 的电阻测量。6517B 还能够测量高…