【Linux操作系统】深入探索Linux进程:创建、共享与管理

进程的创建是Linux系统编程中的重要概念之一。在本节中,我们将介绍进程的创建、获取进程ID和父进程ID、进程共享、exec函数族、wait和waitpid等相关内容。
在这里插入图片描述

文章目录

    • 1. 进程的创建
      • 1.1 函数原型和返回值
      • 1.2 函数示例
    • 2. 获取进程ID和父进程ID
      • 2.1 函数原型和返回值
      • 2.2 函数示例
    • 3. exec函数族
      • 3.1 `exec()`函数族的常见成员:
      • 3.2 函数示例
    • 4. wait和waitpid
      • 4.1 函数解释
      • 4.2 函数示例
  • 总结


1. 进程的创建

在Linux系统中,进程的创建使用fork()系统调用。fork()系统调用会创建一个与当前进程相同的子进程,子进程会复制父进程的所有资源,包括代码、数据和文件描述符等。


1.1 函数原型和返回值

fork()函数的原型如下:

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

fork()函数 没有任何参数 ,它的返回值是一个pid_t类型的整数。具体解释如下:

  • 如果调用成功,fork()函数会在父进程中返回子进程的PID(子进程ID),在子进程中返回0。
  • 如果调用失败,fork()函数会返回-1,并设置errno来指示错误类型。

1.2 函数示例

代码示例如下:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main() {pid_t pid = fork();if (pid == -1) {printf("Failed to fork a new process.\n");return 1;} else if (pid == 0) {printf("This is the child process.\n");} else {printf("This is the parent process.\n");}return 0;
}

上述代码中,fork()系统调用会返回两次,分别在父进程和子进程中返回。通过判断返回值,我们可以区分父进程和子进程,并执行不同的代码逻辑。



2. 获取进程ID和父进程ID

在Linux系统中,可以使用getpid()getppid()系统调用来获取当前进程的ID和父进程的ID。


2.1 函数原型和返回值

getpid()getppid()函数的原型如下:

#include <sys/types.h>
#include <unistd.h>pid_t getpid(void);
pid_t getppid(void);

这两个函数都 没有任何参数 ,它们的返回值都是一个pid_t类型的整数。具体解释如下:

  • getpid()函数返回调用进程的进程ID(PID)。
  • getppid()函数返回调用进程的父进程的进程ID(PPID)。

2.2 函数示例

代码示例如下:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main() {pid_t pid = getpid();pid_t ppid = getppid();printf("Process ID: %d\n", pid);printf("Parent Process ID: %d\n", ppid);return 0;
}


3. exec函数族

在Linux系统中,exec()函数族可以用于将当前进程替换为新的程序。exec()函数族包括execl()execv()execle()execve()等函数。这些函数可以根据不同的参数形式来执行不同的替换方式。


3.1 exec()函数族的常见成员:

  1. int execl(const char *path, const char *arg, ...);

    • 参数 path 是要执行的新程序的路径。
    • 参数 arg 是一个字符串,表示新程序的第一个参数。
    • 可变参数列表是新程序的其他参数,必须以 NULL 结束。
    • 函数执行成功时不会返回,如果返回则表示执行失败。

  2. int execv(const char *path, char *const argv[]);

    • 参数 path 是要执行的新程序的路径。
    • 参数 argv 是一个字符串数组,表示新程序的参数列表,最后一个元素必须是 NULL
    • 函数执行成功时不会返回,如果返回则表示执行失败。

  3. int execle(const char *path, const char *arg, ..., char *const envp[]);

    • 参数 path 是要执行的新程序的路径。
    • 参数 arg 是一个字符串,表示新程序的第一个参数。
    • 可变参数列表是新程序的其他参数,必须以 NULL 结束。
    • 参数 envp 是一个字符串数组,表示新程序的环境变量列表,最后一个元素必须是 NULL
    • 函数执行成功时不会返回,如果返回则表示执行失败。

  4. int execvp(const char *file, char *const argv[]);

    • 参数 file 是要执行的新程序的文件名。
    • 参数 argv 是一个字符串数组,表示新程序的参数列表,最后一个元素必须是 NULL
    • 函数执行成功时不会返回,如果返回则表示执行失败。

这些函数在执行成功时不会返回,而是直接将当前进程替换为新程序。如果返回,则表示执行失败,可以根据返回值来判断错误类型。

exec()函数族可以用于在当前进程中加载和执行新程序,可以实现程序的动态切换和功能扩展。一般情况下,exec()函数族会在调用fork()函数创建子进程后使用,以替换子进程的代码和数据。


3.2 函数示例

代码示例如下:

#include <stdio.h>
#include <unistd.h>int main() {printf("Before exec()\n");execl("/bin/ls", "ls", "-l", NULL);printf("After exec()\n");return 0;
}

上述代码中,execl()函数会将当前进程替换为ls -l命令。execl()函数的第一个参数是要执行的程序路径,后续参数是传递给新程序的命令行参数。



4. wait和waitpid

在Linux系统中,父进程可以使用wait()waitpid()系统调用等待子进程的结束。这些系统调用会阻塞父进程的执行,直到子进程结束。


4.1 函数解释

wait()waitpid()是用于等待子进程结束并获取子进程的退出状态的函数。

  1. pid_t wait(int *status);

    • 函数会挂起当前进程,直到一个子进程结束。
    • 如果成功等到子进程结束,函数会返回子进程的进程ID。
    • 参数 status 是一个指向整型的指针,用于存储子进程的退出状态信息。
    • 如果调用失败,函数会返回-1。

  2. pid_t waitpid(pid_t pid, int *status, int options);

    • 函数会挂起当前进程,直到指定的子进程结束。
    • 参数 pid 指定要等待的子进程的进程ID。
    • 参数 status 是一个指向整型的指针,用于存储子进程的退出状态信息。
    • 参数 options 是一个整型值,用于指定等待的选项。
    • 如果调用失败,函数会返回-1。

wait()waitpid()函数的返回值是子进程的进程ID,如果调用失败,则返回-1。通过参数 status 可以获取子进程的退出状态信息,包括退出码、终止信号等。

waitpid()函数相比于wait()函数更加灵活,可以通过参数 pidoptions 控制等待的子进程。

其中,pid 的取值可以是:

  • -1:等待任意子进程。
  • 0:等待与当前进程组ID相同的子进程。
  • 具体的子进程ID:等待指定的子进程。

options 参数可以通过位掩码的方式指定多个选项,常用的选项有:

  • WNOHANG:非阻塞方式,如果没有子进程结束,立即返回。
  • WUNTRACED:也会返回已经停止的子进程的状态。
  • WCONTINUED:也会返回已经继续运行的子进程的状态。

wait()waitpid()函数可以用于处理子进程的退出状态,释放子进程的资源,并进行进程间的同步。在使用这两个函数时,需要注意处理错误情况和避免僵尸进程的产生。


4.2 函数示例

代码示例如下:

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>int main() {pid_t pid = fork();if (pid == -1) {printf("Failed to fork a new process.\n");return 1;} else if (pid == 0) {printf("This is the child process.\n");} else {wait(NULL);printf("This is the parent process.\n");}return 0;
}

上述代码中,父进程使用wait(NULL)系统调用等待子进程的结束。wait()系统调用会阻塞父进程的执行,直到子进程结束。



总结

  1. fork()函数:用于创建子进程,返回值不同表示在不同的进程中执行。
  2. exec()函数族:用于在当前进程中加载和执行新程序,可以实现程序的动态切换和功能扩展。
    • execl():接受可变参数的形式,参数以字符串形式传递。
    • execle():接受可变参数的形式,同时传递环境变量。
    • execvp():接受参数数组的形式,参数以字符串数组形式传递。
  3. wait()waitpid()函数:用于等待子进程结束并获取子进程的退出状态。
    • wait():等待任意子进程结束。
    • waitpid():可以指定等待的子进程。
    • 通过参数 status 可以获取子进程的退出状态信息。
    • 可以通过 options 参数控制等待的选项,如非阻塞方式等。
    • 需要注意处理错误情况和避免僵尸进程的产生。

这些函数和系统调用可以用于进程的创建、执行和等待,实现进程间的同步和协作。通过这些函数,可以实现进程的动态切换、功能扩展和资源释放。同时,需要注意处理错误情况,避免产生僵尸进程和资源泄漏的问题。

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

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

相关文章

java练习4.快速查找

题目: 数组 arr[6,1,3,7,9,8,5,4,2],用快速排序进行升序排序. import java.util.Random;public class recursionDemo {public static void main(String[] args) {/*快速排序:* 第一轮:以0索引为基准数,确定基准数在数组正确的位置,* 比基准数小的放到左边,比基准数大的放在右边…

Markdown 入门基础

文章目录 Markdown 是什么&#xff1f;为什么要使用 Markdown?工欲善其事&#xff0c;必先利其器Markdown 的工作原理Markdown 有什么用&#xff1f;网站文件资料笔记书籍演示文稿邮件文档 Markdown 方言其它资源 Markdown 是什么&#xff1f; Markdown 是一种轻量级的标记语…

【c语言】通讯录(动态版+文件+背景音乐)含源码

开饭了&#xff0c;之前写的通讯录&#xff0c;是否会有人觉得申请1000人的空间是不是有点用不上呀&#xff0c;怎么才能做到要多少申请多少个呢&#xff1f;&#xff1f;我们学完动态内存管理&#xff0c;和文件的相关操作&#xff0c;终于可以继续完善我们的通讯录了 船新版本…

(搜索) 剑指 Offer 12. 矩阵中的路径 ——【Leetcode每日一题】

❓剑指 Offer 12. 矩阵中的路径 难度&#xff1a;中等 给定一个 m * n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 单词必须按照字母顺序&#xff0c;通过相邻的单元格内的字母构…

UE4/5数字人MetaHuman的控制绑定资产使用

开始操作 首先我们创建一个关卡序列&#xff1a; 打开后将我们的数字人放进去【右键&#xff0c;第一个添加进去】&#xff1a; 我们会自动进入动画模式&#xff0c;没有的话&#xff0c;就自己进入一下&#xff0c; 然后我们去寻找我们的控制绑定资产。 找到控制绑定资产 …

牛客网华为OD前端岗位,面试题库练习记录01

题目一 质数因子 功能:输入一个正整数&#xff0c;按照从小到大的顺序输出它的所有质因子&#xff08;重复的也要列举&#xff09;&#xff08;如180的质因子为2 2 3 3 5 &#xff09; JavaScript Node ACM模式 const rl require("readline").createInterface({ i…

设计HTML5表格

在网页设计中&#xff0c;表格主要用于显示包含行、列结构的二维数据&#xff0c;如财务表格、调查数据、日历表、时刻表、节目表等。在大多数情况下&#xff0c;这类信息都由列标题或行标题及数据构成。本章将详细介绍表格在网页设计中的应用&#xff0c;包括设计符合标准化的…

(7)(7.3) 自动任务中的相机控制

文章目录 前言 7.3.1 概述 7.3.2 自动任务类型 7.3.3 创建合成图像 前言 本文介绍 ArduPilot 的相机和云台命令&#xff0c;并说明如何在 Mission Planner 中使用这些命令来定义相机勘测任务。这些说明假定已经连接并配置了相机触发器和云台(camera trigger and gimbal hav…

我的编程语言学习笔记

前言 作为一名编程初学者&#xff0c;我深知学习编程需要不断积累和记录。在这篇博客文章中&#xff0c;我将分享一些我在学习C/C编程语言过程中记录的常用代码、特定函数、复杂概念以及特定功能。希望能与大家一起切磋进步&#xff01; 常用代码&#xff1a; 1. 输入输出操作…

Algorithem Review 5.2 图论

网络流 设源点为 s s s&#xff0c;汇点为 t t t&#xff0c;每条边 e e e 的流量上限为 c ( e ) c(e) c(e)&#xff0c;流量为 f ( e ) f(e) f(e)。割 指对于某一顶点集合 P ⊂ V P \subset V P⊂V&#xff0c;从 P P P 出发指向 P P P 外部的那些原图中的边的集合&a…

【Docker报错】docker拉取镜像时报错:no such host

报错信息 [rootSoft soft]# docker pull mysql Using default tag: latest Error response from daemon: Head "https://registry-1.docker.io/v2/library/mysql/manifests/latest": dial tcp: lookup registry-1.docker.io on 192.168.80.2:53: no such host解决方法…

【【典型电路设计之片内存储器的设计之RAM的Verilog HDL描述一】】

典型电路设计之片内存储器的设计之RAM的Verilog HDL描述一 RAM是随机存储器&#xff0c;存储单元的内容可按需随意取出或存入。这种存储器在断电后将丢失所有数据&#xff0c;一般用来存储一些短时间内使用的程序和数据。 其内部结构如下图所示&#xff1a; 例&#xff1a;用…

docker简介

目录 docker简介 1.什么是docker 2.基本结构 3.docker优势 4.docker改变了什么 5.docker三大基本概念 1.docker镜像 2.容器 3.仓库 docker简介 1.什么是docker Docker 是一个开源项目&#xff0c; 诞生于 2013 年初&#xff0c;最初是 dotCloud 公司内部的一个业余项目。…

matlab保存图片

仅作为记录&#xff0c;大佬请跳过。 文章目录 用界面中的“另存为”用saveas 用界面中的“另存为” 即可。 参考 感谢大佬博主文章&#xff1a;传送门 用saveas 必须在编辑器中的plot之后用saveas&#xff08;也就是不能在命令行中单独使用——比如在编辑器中plot&#xf…

GuLi商城-前端基础Vue-指令-单向绑定双向绑定

什么是指令? 指令 (Directives) 是带有 v- 前缀的特殊特性。 指令特性的预期值是:单个 JavaScript 表达式。 指令的职责是&#xff0c;当表达式的值改变时&#xff0c;将其产生的连带影响&#xff0c;响应式地作用于DOM 例如我们在入门案例中的 v-on&#xff0c;代表绑定事…

Docker+Selenium Grid搭建自动化测试平台

安装docker yum install -y yum-utils device-mapper-persistent-data lvm2 yum-config-manager –add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo yum install docker-ce -y Create a Docker Network docker network create grid 下载镜像 hu…

蔡司关注全民运动眼健康:与蔡司智锐镜片KEEP住视力健康

众所周知&#xff0c;运动是对我们身体最大的投资&#xff0c;但是对于视力有问题的消费者来说&#xff0c;不合适的眼镜无疑是运动路上的绊脚石&#xff0c;跑步运动时眼镜总是往下掉&#xff0c;不仅没有相对稳定的视野&#xff0c;还特别没安全感&#xff0c;由此可见一副优…

【云原生、k8s】Calico网络策略

第四阶段 时 间&#xff1a;2023年8月17日 参加人&#xff1a;全班人员 内 容&#xff1a; Calico网络策略 目录 一、前提配置 二、Calico网络策略基础 1、创建服务 2、启用网络隔离 3、测试网络隔离 4、允许通过网络策略进行访问 三、Calico网络策略进阶 1、创…

【数据结构OJ题】环形链表II

原题链接&#xff1a;https://leetcode.cn/problems/linked-list-cycle-ii/description/ 1. 题目描述 2. 思路分析 如果链表存在环&#xff0c;则fast和slow会在环内相遇&#xff0c;定义相遇点到入口点的距离为X&#xff0c;定义环的长度为C&#xff0c;定义头到入口的距离为…

CentOS安装Docker(超详细)

文章目录 1.CentOS安装Docker1.1.卸载&#xff08;可选&#xff09;1.2.安装docker1.3.启动docker1.4.配置镜像加速 2.CentOS安装DockerCompose2.1.下载2.2.修改文件权限2.3.Base自动补全命令&#xff1a; 3.Docker镜像仓库3.1.简化版镜像仓库3.2.带有图形化界面版本3.3.配置Do…