操作系统笔记之进程调用API中的getpid、fork、wait、exec补充

操作系统笔记之进程调用API中的getpid、fork、wait、exec补充

code review!

效果图

—— 杭州 2024-03-17 夜

文章目录

  • 操作系统笔记之进程调用API中的getpid、fork、wait、exec补充
    • 1.getpid()
    • 2.fork()
    • 3.wait()
    • 4.exec()
    • 5.通常,exec() 调用与 fork() 调用一起使用,为什么?
      • `fork()` 的作用
      • `exec()` 的作用
      • `fork()` 和 `exec()` 一起使用的原因
      • 实例
    • 6.fork() 和 exec() 一起使用时,子进程的调用会返回吗?

1.getpid()

getpid() 是一个在 Unix-like 系统(比如 Linux 和 macOS)中常用的系统调用函数,它用于获取当前进程的进程标识符(Process ID,简称 PID)。该函数定义在 <unistd.h> 头文件中,属于 POSIX 标准的一部分。

每个运行中的进程都有一个唯一的 PID,这是一个非负整数。这个标识符可以用于控制进程,比如发送信号给进程来终止它或者查询进程的状态。

函数原型

getpid() 函数的原型如下:

#include <unistd.h>pid_t getpid(void);

这里 pid_t 通常是 int 类型的别名,用于表示进程 ID。

返回值

getpid() 函数返回调用进程的 PID。因为每个进程都有一个唯一的 PID,并且 getpid() 不会失败,所以它没有失败的返回值。

示例代码

以下是一个简单的示例,演示了如何在 C 程序中调用 getpid() 函数:

#include <stdio.h>
#include <unistd.h>int main() {pid_t pid = getpid(); // 调用 getpid() 获取当前进程的 PIDprintf("The Process ID (PID) is: %d\n", pid);return 0;
}

当你编译并运行这个程序时,它会输出当前进程的 PID。

应用场景

getpid() 在多种情况下可能会被使用:

  • 日志记录:在进行系统日志记录时,记录当前进程的 PID 可以帮助在出现问题时追踪到具体的进程。
  • 进程管理:脚本或程序可能需要知道其自身的 PID,以便于创建锁文件,防止多个实例同时运行。
  • 调试:在调试多进程程序时,知道不同进程的 PID 可以帮助区分它们的输出或行为。
  • 信号处理:发送信号给特定进程时需要知道其 PID。

注意事项

  • 在 Unix-like 系统中,PID 1 通常是初始化进程(init 或 systemd),它是所有其他用户空间进程的祖先。
  • PID 是一个有限资源,在长时间运行的系统中可能会耗尽。不过,系统会在 PID 耗尽时回收和重用旧的、不再使用的 PID。
  • 在多线程程序中,所有线程共享同一个 PID,因为它们运行在同一个进程上下文中。如果你需要获取线程的唯一标识符,应该使用 pthread_self() 或其他相关函数。

2.fork()

fork() 是一个用于创建进程的系统调用。理解 fork() 的关键是要知道它会创建一个与原始(父)进程几乎完全相同的新进程(子进程)。让我们用一个更详细的方式来说明这个过程:

  1. 调用 fork(): 当一个进程(我们称之为父进程)执行到 fork() 系统调用时,它会请求操作系统创建一个新的进程。

  2. 创建子进程: 操作系统会复制父进程的整个状态到子进程中。这意味着子进程将获得父进程数据段、代码段和堆栈的副本。在许多操作系统中,这种复制是通过写时复制(Copy-on-Write, COW)机制实现的,以提高效率。实际的内存页只有在其中一个进程尝试写入时才会被复制。

  3. 区分父子进程: fork() 调用在父进程中返回子进程的 PID,在子进程中返回 0。这是父进程和子进程的代码可以区分两个进程的关键所在。

  4. 独立执行: 一旦 fork() 完成,两个进程(父进程和子进程)都将从 fork() 调用之后的指令开始独立执行。这两个进程有各自独立的地址空间,所以一个进程对内存的改变不会影响另一个进程。

  5. 资源共享: 子进程会继承父进程的文件描述符。这些文件描述符指向相同的文件表项,意味着父子进程可以共享打开的文件等资源。

  6. 独立生命周期: 子进程有自己的生命周期,它可以独立于父进程执行,也可以执行不同的代码。通常,子进程会调用 exec() 系列函数来替换自己的内存空间,包括代码和数据,以运行一个新的程序。

简单的 fork() 示例

让我们看一个 fork() 的示例,以便更好地理解这个过程:

在这里插入图片描述

运行:
在这里插入图片描述

代码

#include <stdio.h>
#include <unistd.h>int main() {printf("Before fork()\n");pid_t pid = fork();if (pid == -1) {// fork失败perror("fork");return 1;} else if (pid > 0) {// 父进程printf("I am the parent process. My PID is %d and my child's PID is %d.\n", getpid(), pid);} else {// 子进程printf("I am the child process. My PID is %d.\n", getpid());}printf("This is the end of the process with PID %d.\n", getpid());return 0;
}

在这个程序中,我们首先打印 “Before fork()”,然后调用 fork()。在 fork() 之后,我们有两个独立的进程:父进程和子进程。每个进程都会执行相应的 if 分支,并且都会打印 “This is the end of the process with PID …”。因此,你会看到 “Before fork()” 只打印一次,而 “This is the end of the process with PID …” 会打印两次,一次用父进程的 PID,一次用子进程的 PID。

3.wait()

wait() 系统调用在 UNIX 和类 UNIX 操作系统中用于使一个父进程等待其子进程结束或改变状态。当子进程结束或停止时,父进程可以通过 wait() 系统调用来收集子进程的退出状态信息。这是一种进程间通信的方式,也是父进程管理子进程生命周期的一种方法。

功能

wait() 提供以下功能:

  • 收集子进程状态: 父进程可以获取子进程的终止状态,例如子进程的退出代码。
  • 回收资源: 当子进程结束时,操作系统会保留一些资源(如进程描述符和统计信息),直到父进程通过 wait() 调用释放。这一步骤被称为 “回收” 子进程。
  • 同步: wait() 可以用来确保父进程在子进程结束之前不会继续执行,实现父子进程间的同步。

使用方法

wait() 调用通常在父进程中这样使用:

#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdio.h>int main() {pid_t pid = fork();if (pid == -1) {// fork失败perror("fork");return 1;} else if (pid > 0) {// 父进程int status;waitpid(pid, &status, 0); // 父进程等待子进程结束if (WIFEXITED(status)) {printf("Child exited with status %d\n", WEXITSTATUS(status));}} else {// 子进程// 执行一些工作..._exit(42); // 子进程结束,返回42作为退出状态}return 0;
}

参数和返回值

wait() 函数的原型如下:

pid_t wait(int *status);
  • status: 这是一个指向整数的指针,用于存储子进程的退出状态。通过宏(比如 WIFEXITEDWEXITSTATUS)可以分析这个状态值。
  • 返回值: 返回子进程的 PID,或在错误时返回 -1。

状态宏

wait() 通过 status 参数传递的状态值可以用下列宏进行分析:

  • WIFEXITED(status): 如果子进程正常结束,则此宏返回真(非0值)。
  • WEXITSTATUS(status): 如果 WIFEXITED 非零,返回子进程的退出状态(即 main() 的返回值或 _exit 的参数)。
  • WIFSIGNALED(status): 如果子进程因为信号而结束,则此宏返回真。
  • WTERMSIG(status): 如果 WIFSIGNALED 非零,返回导致子进程终止的信号编号。
  • WIFSTOPPED(status): 如果子进程处于停止状态,则此宏返回真。
  • WSTOPSIG(status): 如果 WIFSTOPPED 非零,返回导致子进程停止的信号编号。

注意事项

  • 如果父进程没有调用 wait(),而子进程已经结束,子进程将变成僵尸进程(Zombie),直到其父进程结束或为其调用 wait()
  • 如果父进程结束而子进程仍在运行,子进程将被 init 进程(PID 为 1)收养,init 将负责调用 wait() 收集状态信息。

wait() 还有几个相关函数,如 waitpid()waitid()wait3()/wait4(),它们提供了更多控制选项,比如非阻塞等待或等待特定的子进程。

4.exec()

当你运行一个程序时,操作系统为该程序创建了一个进程。每个进程都有自己的内存空间,其中包含了运行程序所需的指令和数据。exec() 系统调用的功能是在一个已经存在的进程中启动一个新的程序。这意味着 exec() 实际上是用一个全新的程序来替换当前进程的内存空间内容。

exec() 是一组函数,不仅仅只有一个。这组函数包括 execl(), execv(), execlp(), execvp(), execle(), execve() 等。这些函数的区别在于它们如何接收参数(比如直接传递参数列表或是通过数组传递参数),以及它们是否搜索系统的 PATH 环境变量来找到可执行文件。

以下是 exec() 函数族中一个函数的典型用法:

#include <stdio.h>
#include <unistd.h>int main() {char *args[] = {"/bin/ls", "-l", NULL}; // 定义了要执行的命令和参数列表execv("/bin/ls", args); // 使用 execv 来执行/bin/ls程序// 如果execv执行成功,以下的代码不会被执行,// 因为当前进程的内存已经被ls程序替换。perror("execv"); // 如果 execv 失败,则打印错误消息return 1;
}

在这个例子中,execv() 被用来在当前进程中运行 /bin/ls 命令。如果 execv() 成功执行,当前的程序(这个示例中的 C 程序)就会停止运行,因为它的内存空间被 ls 命令的代码和数据所替换。因此,程序中 execv() 调用之后的代码(在这里是 perror()return 1)不会被执行。

如果 execv() 调用失败了(比如,如果 /bin/ls 不是一个有效的可执行文件),则 execv() 会返回 -1,并且 errno 会被设置为描述错误的代码。在这种情况下,perror() 会被执行,它会根据 errno 的值打印一条错误消息。

通常,exec() 调用与 fork() 调用一起使用,这样可以先通过 fork() 创建一个新的子进程,然后在子进程中使用 exec() 来替换为另一个程序。父进程可以继续执行其他任务,或者等待子进程完成。

5.通常,exec() 调用与 fork() 调用一起使用,为什么?

fork() 的作用

  • 创建进程: fork() 创建了一个新的子进程,这个子进程几乎是父进程的完整副本。它有自己的进程ID,并且复制了父进程的内存布局。
  • 独立运行: 一旦 fork() 成功,你就有了两个几乎相同的进程:一个父进程和一个子进程。

exec() 的作用

  • 替换程序: exec() 家族的函数用于在一个进程中启动一个新程序。它替换当前进程的内存空间,包括代码和数据。
  • 执行新任务: exec() 通常在 fork() 创建的子进程中调用,以确保子进程执行的是与父进程不同的新任务。

fork()exec() 一起使用的原因

  • 保护父进程: 如果只使用 exec(),你将失去当前正在运行的程序,因为它会被新程序替换。fork() 允许父进程继续运行,同时子进程可以去执行新任务。
  • 执行并发任务: 使用 fork()exec(),你可以让父进程和子进程同时(并发地)执行不同的任务。

实例

想象你有一个程序,就像一个简单的命令行界面。用户输入一个命令,比如 ls

  1. 程序调用 fork() 创建一个新的子进程。
  2. 子进程使用 exec() 来执行 ls 命令。
  3. 父进程等待子进程执行完毕。

在这个过程中:

  • 如果没有 fork(),原有的程序就会停止运行来执行 ls
  • 如果只有 fork() 而没有 exec(),子进程将会做和父进程完全相同的事情,这不是我们想要的。

结合使用 fork()exec() 允许父进程继续运行自己的代码,而子进程则运行一个全新的程序。这是多任务操作系统中常见的操作,它允许系统同时处理多个任务。

6.fork() 和 exec() 一起使用时,子进程的调用会返回吗?

fork()exec() 被一起使用时,子进程的行为取决于 exec() 调用是否成功:

  1. 如果 exec() 调用成功

    • 子进程的 exec() 调用不会返回,因为子进程的原始程序代码已经被新程序替换。子进程从此开始执行新程序的代码,之前的执行上下文(包括调用 exec() 的代码)不复存在。
    • 子进程将继续作为新程序运行,直到该程序结束或遇到错误。
  2. 如果 exec() 调用失败

    • 子进程中的 exec() 调用会返回 -1,并且 errno 会被设置为描述错误的代码。在这种情况下,子进程通常会执行一些错误处理的代码,例如打印出错信息,并且随后通常会立即退出。
    • 子进程在 exec() 调用失败后通常会调用 exit()_exit() 函数来结束自身,因为子进程的正常逻辑是执行另一个程序,如果这一步骤失败了,通常就没有理由继续执行原来的程序代码了。

简而言之,如果 exec() 成功,子进程不会返回到原程序代码;如果 exec()失败,子进程会返回一个错误,而且通常会紧接着退出。父进程可以通过 wait()waitpid()调用来监测子进程的退出状态,以了解子进程是正常结束还是遇到了错误。

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

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

相关文章

002_avoid_for_loop_in_Matlab避免使用for循环

避免使用for循环 在程序设计思想中&#xff0c;循环是一个很有力的工具。在循环中&#xff0c;计算机很轻松地重复执行相同的操作。循环是汇编之上的编程中最重要的概念之一。Matlab的循环有两个语言构造&#xff0c;一个是for循环&#xff0c;另一个是while循环。在Matlab中&…

JetBrains全家桶激活,分享 GoLand 2024 激活的方案

大家好&#xff0c;欢迎来到金榜探云手&#xff01; GoLand 公司简介 JetBrains 是一家专注于开发工具的软件公司&#xff0c;总部位于捷克。他们以提供强大的集成开发环境&#xff08;IDE&#xff09;而闻名&#xff0c;如 IntelliJ IDEA、PyCharm、和 GoLand等。这些工具被…

总结mac下解决matplotlib中文显示问题的几种方法

一、前言&#xff1a; 使⽤matplotlib画图时&#xff0c;由于matplotlib默认没有中⽂&#xff0c;显⽰中文时会出现空⽩⼩⽅块。 二、方法&#xff1a; 2.1 matplotlib中使用SimHei字体 1&#xff09;进入终端后查看matplotlib的字体路径&#xff1a; $ python >>&g…

Linux学习-进程

目录 进程基本概念 进程相关命令 进程的创建 进程的调度 进程相关函数接口 进程的消亡 实例&#xff1a;创建九个子进程 目录 进程基本概念 进程相关命令 进程的创建 进程的调度 进程相关函数接口 进程的消亡 实例&#xff1a;创建九个子进程 exec函数…

Python脚本:用py处理PDF的五大功能

一、代码 【第三方库】3个 【Py版本】3.9 【使用前提】关闭所有的word文档 import os from datetime import datetime from docx2pdf import convert from pdf2docx import parse from PyPDF2 import PdfMerger from PyPDF2 import PdfReader,PdfWriter#将文件夹中的所有Wo…

C++语言学习(二)—— C++语言的基本知识

目录 一、面向对象的三个核心概念 二、C语言中的I/O口 三、C语言中的数据类型​​​​​​​ 3.1 逻辑类型 3.2 引用类型 3.2.1 引用作为函数参数 3.2.2 引用作为函数返回值 3.2.3 引用作为类成员 3.3 类类型 四、 C语言中的内联函数 五、 函数重载 六、 带默认形参…

四、分布式锁之自定义分布式锁

1、基本原理和实现方式对比 分布式锁&#xff1a;满足分布式系统或集群模式下多个进程可见并且互斥的锁。分布式锁的核心思想就是多线程都使用同一把锁&#xff0c;实现程序串行执行。 分布式锁需要具备的条件&#xff1a; 特性含义可见性多个线程都能感知到变化互斥性分布…

数据库系统概论-第16章 数据仓库与联机分析处理技术

概念性的介绍&#xff0c;一略而过&#xff0c;不重要。 16.1 数据仓库技术 16.2 联机分析处理技术 16.3 数据挖掘技术 16.4 大数据时代的新型数据仓库 16.5 小结

G - Find a way

题目分析 1.双重bfs,遍历两个起点求最短路再计算总和即可 2.唯一的坑点在于对于一个KFC&#xff0c;两人中可能有一个到不了&#xff0c;所以还要对到不了的点距离做处理 #include <bits/stdc.h> using namespace std; using ll long long; const int N 220;struct pos…

Linux/Ubuntu/Debian从控制台启动程序隐藏终端窗口

如果你想从终端运行应用程序但隐藏终端窗口. 你可以这样做&#xff1a; 在后台运行&#xff1a; 你只需在命令末尾添加一个与号 (&) 即可在后台运行它。 例如&#xff1a; your_command &将 your_command 替换为你要运行的命令。 这将在后台启动该命令&#xff0c…

Three.js基础入门介绍——【毕业季】Three.js动态相册

前言 岁月匆匆&#xff0c;又是一年毕业季&#xff0c;这次做个动态相册展示图片&#xff0c;放些有意思的内容&#xff0c;一起回忆下校园生活吧。 预期效果 相册展示和点选切换&#xff0c;利用相机旋转和移动来实现一个点击切图平滑过渡的效果。 实现流程 基本流程 1、搭…

【python】python汽车效能数据集—回归建模(源码+数据集)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

【学习】软件测试行业未来的发展趋势预测

近年来&#xff0c;随着中国数字经济的蓬勃发展&#xff0c;软件测试行业也迎来了新的春天。从早期的手工测试到自动化测试&#xff0c;再到持续集成和持续交付&#xff0c;中国的软件测试行业经历了快速的发展和变革。各行各业均对软件测试提出了更高的要求&#xff0c;尤其在…

将数据转换成xml格式的文档并下载

现在有一个实体类对象的集合&#xff0c;需要将它们转换为xml文档&#xff0c;xml文档就是标签集合的嵌套&#xff0c;例如一个学生类&#xff0c;有姓名、年龄等&#xff0c;需要转换成一下效果&#xff1a; <student><age>14</age><name>张三</na…

【Java】Oracle发布Java22最新版本

甲骨文&#xff08;ORACLE&#xff09;已经于2023年3月19日正式发布了最新版本的JDK&#xff0c;版本号&#xff1a;22 根据官方声明&#xff0c;Java 22 (Oracle JDK 22) 在性能、稳定性和安全性方面进行了数千种改进&#xff0c;包括对Java 语言、其API 和性能&#xff0c;以…

docker 哲学 - 网络桥接器、容器网络接口 、容器间的通信方式

1、解释 docker0 veth eth 2、vethXX 和 ethXX 是肯定一一对应吗 比如 eth1 对应 veth1 3、如果 A容器使用 默认创建方式 。定义他内部网络为 eth0&#xff0c;容器B使用 --network 连上 已创建的网络 172.89.2.1 。此时假设 B的 ip是 172.89.2.2 &#xff0c;容器网络接口是 e…

Godot 学习笔记(4):一切以场景为中心

文章目录 前言场景搭建新建子场景最简单的按钮事件 手动控制场景手动加载场景添加多个场景对象更快速的获取脚本对象 删除多个场景对象脚本命名的问题 总结 前言 Godot的场景是C#与Godot最后的中间连接。我们解决了场景的加载&#xff0c;我们基本可以保证C#和godot之间的彻底…

C++初阶:vector相关练习

目录 1. 只出现一次的数2. 杨辉三角3. 删除有序数组中的重复项4. 只出现一次的数II5. 只出现一次的数III6. 数组中出现次数超过一半的数7. 电话号码的字母组合&#xff08;多叉树遍历&#xff09; 1. 只出现一次的数 题目信息&#xff1a; 题目链接&#xff1a; 只出现一次的数…

工程信号的去噪和(分类、回归和时序)预测

&#x1f680;【信号去噪及预测论文代码指导】&#x1f680; 还为小论文没有思路烦恼么&#xff1f;本人专注于最前沿的信号处理与预测技术——基于信号模态分解的去噪算法和深度学习的信号&#xff08;回归、时序和分类&#xff09;预测算法&#xff0c;致力于为您提供最精确、…

ruoyi-nbcio-plus基于vue3的flowable增加开始节点的表单绑定修改

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 http://122.227.135.243:9666/ 更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a…