OS | 第5章 插叙:进程API

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 countprintf("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 fileclose(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 countprintf("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.rhkb.cn/news/121200.html

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

相关文章

IDEA报错:Plugin ‘org.springframework.boot:spring-boot-maven-plugin:‘ not found

问题&#xff1a; 使用IDEA新建spring boot项目&#xff0c;报错如下&#xff1a; Plugin org.springframework.boot:spring-boot-maven-plugin: not found解决办法&#xff1a; 1.在本地maven仓库中找到spring-boot-maven-plugin的版本号 2.在pom.xml文件中添加对应的版本…

城市小车的优势,用五菱宏光mini,轻松应对城市拥堵与环保挑战。

掌握五菱宏光mini的驾驶技巧&#xff0c;让拥堵不再困扰你 合理利用车辆尺寸&#xff0c;轻松穿梭于城市道路 五菱宏光mini的尺寸小巧&#xff0c;长度不到3米&#xff0c;宽度不到1.5米&#xff0c;让你可以在狭窄的城市街道上轻松穿梭。掌握这一技巧&#xff0c;让你在拥堵…

什么是瓷片电容封装 | 百能云芯

瓷片电容封装是一种常见的电子元件封装方式&#xff0c;它广泛应用在电子设备中&#xff0c;用于存储和释放电荷&#xff0c;以实现电路的稳定工作。在本文中&#xff0c;我们将详细介绍瓷片电容封装的特点以及用途。 瓷片电容封装的特点&#xff1a; 瓷片电容是一种以陶瓷材料…

【USRP】调制解调系列6:16APSK、32APSK 、基于labview的实现

APSK APSK是&#xff0c;与传统方型星座QAM&#xff08;如16QAM、64QAM&#xff09;相比&#xff0c;其分布呈中心向外沿半径发散&#xff0c;所以又名星型QAM。与QAM相比&#xff0c;APSK便于实现变速率调制&#xff0c;因而很适合目前根据信道及业务需要分级传输的情况。当然…

机器学习前沿:改进自身缺陷,满足新战略

前机械师&#xff08; 来源) 一、说明 机器学习在人工智能历史上扮演重要角色&#xff0c;然而&#xff0c;存在问题也不少。为了适应新时代和新任务&#xff0c;不做出重大改进是不可能的&#xff0c;本篇就一些突出问题和改进做出讨论。以便读者掌握未来的思路和方向。 二、机…

1783_CMD启动MATLAB同时执行一个脚本

全部学习汇总&#xff1a; GitHub - GreyZhang/g_matlab: MATLAB once used to be my daily tool. After many years when I go back and read my old learning notes I felt maybe I still need it in the future. So, start this repo to keep some of my old learning notes…

框架分析(9)-Hibernate

框架分析&#xff08;9&#xff09;-Hibernate 专栏介绍Hibernate特性对象关系映射&#xff08;ORM&#xff09;数据库连接和事务管理查询语言&#xff08;HQL&#xff09;缓存机制透明的持久化操作对象的延迟加载事务管理 优缺点优点简化数据库操作跨数据库平台高度可定制性缓…

两台电脑共享文件设置

步骤一&#xff1a;确保网络连接正常&#xff0c;可网线直连。 两台电脑IP设置&#xff0c;例&#xff1a; 步骤二&#xff1a;启用共享功能。 1.在【控制面板】中选择【网络和Internet】&#xff1b; 2.点击【网络和共享中心】&#xff0c;在左侧导航栏中&#xff0c;点击【…

1775_树莓派3B键盘映射错误解决

全部学习汇总&#xff1a; GitHub - GreyZhang/little_bits_of_raspberry_pi: my hacking trip about raspberry pi. 入手树莓派3B之后用了没有多长时间&#xff0c;最初的这段时间感觉想让它代替我的PC机是不肯能的。性能先不说&#xff0c;我完全没有找到当初在我的笔记本上使…

【STM32】IIC使用中DMA传输时 发送数据总少一个的问题

问题描述 在使用STM32 I2C数据发送过程中&#xff0c;发现每轮实际发送出去的数据总比在DMA配置中设定的传输数据个数要少一个。比方说&#xff1a;DMA配置里设定的传输数据个数是10个&#xff0c;结果发现在总线上只能发出9个&#xff0c;经过进一步发现是少了最后一个数据。…

ARM DIY(六)音频调试

前言 今天&#xff0c;调试一下音频 硬件焊接 硬件部分核心是 LM4871 音频功放芯片 对于 SOC 来讲很简单&#xff0c;就一个引脚 HPOUTL&#xff08;单声道&#xff09;&#xff1b;对于扬声器来讲也很简单&#xff0c;就两个引脚&#xff0c;插上就可以了。 另外一个关键点…

实现 Trie (前缀树)

题目链接 实现 Trie (前缀树) 题目描述 注意点 word 和 prefix 仅由小写英文字母组成 解答思路 首先要理解前缀树是什么&#xff0c;参照该篇文章【图解算法】模板变式——带你彻底搞懂字典树(Trie树)在了解前缀树是什么后&#xff0c;设计前缀树就会更加容易&#xff0c;…

【B树 B+树】B树、B+树理论

目录 引入B树B树定义和性质m阶B树核心特性 B树B树的查找 引入B树 满足上面两个策略就是B树&#xff1a; m 叉查找树中&#xff0c;规定除了根节点外&#xff0c;任何结点至少有 ⌈ m / 2 ⌉ \lceil m/2 \rceil ⌈m/2⌉ 个分叉&#xff0c;即至少含有 ⌈ m / 2 ⌉ \lceil m/2 \…

Calico IP In IP模拟组网

Calico IP In IP模拟组网 网络架构 模拟组网 先在k8s-master-1节点执行如下命令&#xff1a; # 创建veth-pair设备对ip link add veth1 type veth peer name eth0# 创建ns1网络命名空间ip netns add ns1# 将eth0网卡插入ns1网络命名空间ip link set eth0 netns ns1# 为ns1网…

08-Vue基础之组件

个人名片&#xff1a; &#x1f60a;作者简介&#xff1a;一名大二在校生 &#x1f921; 个人主页&#xff1a;坠入暮云间x &#x1f43c;座右铭&#xff1a;懒惰受到的惩罚不仅仅是自己的失败&#xff0c;还有别人的成功。 &#x1f385;**学习目标: 坚持每一次的学习打卡 文章…

对话永洪科技CEO何春涛:专注BI,决胜AI时代丨数据猿专访

大数据产业创新服务媒体 ——聚焦数据 改变商业 大数据、云计算、人工智能为代表的新一代信息技术走向普及&#xff0c;数据驱动业务&#xff0c;逐渐成为现代化企业管理、运作的日常。对于年均复合增长率超过20%的国内商业智能&#xff08;BI&#xff09;市场而言&#xff0c…

【SpringSecurity】十一、SpringSecurity集成JWT实现token的方法与校验

文章目录 1、依赖与配置2、JWT工具类3、认证成功处理器4、创建JWT过滤器5、安全配置类 1、依赖与配置 添加JWT的maven依赖&#xff1a; <!-- 添加jwt的依赖 --> <dependency><groupId>com.auth0</groupId><artifactId>java-jwt</artifactId…

多线程场景下谨慎使用@Transactional注解,你不信我也没办法

最近遇到一个很诡异的bug&#xff0c;觉得很有趣也很值得分享&#xff0c;于是想写篇文章记录下来&#xff0c;希望有缘人看到以后少踩坑~ 先简单说下场景&#xff1a;有个任务平台&#xff0c;功能很多但我们只关注 提交任务和取消任务 两个功能&#xff0c;并且取消任务后会有…

K210-CanMV IDE开发软件

K210-CanMV IDE开发软件 界面功能简介连接设备临时运行开机运行程序 界面功能简介 区域①菜单栏&#xff1a;操作文件&#xff0c;使用工具等。 区域②快捷按钮&#xff1a;区域①中的文件和编辑中部分功能的快捷方式。 区域③连接设备&#xff1a;连接设备和程序控制按钮。 …

【网络安全带你练爬虫-100练】第20练:数据处理-并写入到指定文档位置

目录 一、目标1&#xff1a;解码去标签 二、目标2&#xff1a;提取标签内内容 三、目标3&#xff1a;处理后的数据插入原位置 四、目标4&#xff1a;将指定的内容插入指定的位置 五、目标5&#xff1a;设置上下文字体格式 六、目标6&#xff1a;向多个不同位置插入不同的…