Linux系统编程_8_进程控制之fork_wait_waitpid函数

2023-11-03 20:32

本文主要是介绍Linux系统编程_8_进程控制之fork_wait_waitpid函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

fork函数:

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

fork用来创建一个子进程;


特点

fork调用后会返回两次,子进程返回0,父进程返回子进程的进程ID;fork返回后,子进程和父进程都从fork函数的下一条语句开始执行;

注意

fork之后,操作系统会复制一个与父进程完全相同的子进程,虽说是父子关系,但是在操作系统看来,他们更像兄弟关系,这两个进程共享代码空间,但是数据空间是互相独立的,子进程数据空间中的内容是父进程的完整拷贝,指令指针也完全相同,子进程拥有父进程当前运行到的位置(两进程的程序计数器pc值相同,也就是说,子进程是从fork返回处开始执行的),但有一点不同,如果fork成功,子进程中fork的返回值是0,父进程中fork的返回值是子进程的进程号,如果fork不成功,父进程会返回错误。
    可以这样想象,2个进程一直同时运行,而且步调一致,在fork之后,他们分别作不同的工作,也就是分岔了。这也是fork为什么叫fork的原因。至于子进程和父进程哪个先执行,这是不确定的,取决于操作系统。
如果用vfork,则可以保证子进程先运行 完成后父进程在运行。

    上面的注意中我们知道,子进程数据空间中的内容是父进程的完整拷贝,就是说子进程中对数据的操作是不会影响父进程的,下面的例子可以说明这一个特点:

#include <stdio.h>
#include <unistd.h>int main()
{int i = 10;pid_t pid;printf("Father's pid:%d\n", getpid());pid = fork();if(pid < 0){perror("fork failure!");return -1;}else if(pid == 0){while(1){i++;printf("Child's i = %d\n", i);sleep(1);}}else{printf("Child's pis:%d\n", pid);while(1){printf("Father's i = %d\n", i);sleep(1);}sleep(1);}return 0;
}
运行结果:

Father's pid:12148
Child's pis:12149
Father's i = 10
Child's i = 11
Father's i = 10
Child's i = 12
Father's i = 10
Child's i = 13

........


还有一点要注意,如果父进程中打开了文件,即内核给应用程序返回一个文件描述符,子进程和父进程的文件描述符所对应的文件表项是共享的,这意味着子进程对文件的读写直接影响父进程的文件位移量(反之同理)。

进程中调用fd2 = dup(fd1) 产生的新的fd2所指向的文件表项和fd1指向的文件表项是相同的;



wait和waitpid函数:

wait和waitpid用来等待子进程结束;

    如果没有子进程,则wait出错返回;

    有子进程,子进程正在运行,则阻塞,等待子进程结束;

    如果子进程已经结束,则得到结束的子进程的信息,并返回;

为什么要用wait和waitpid函数?

    如果父进程先结束,子进程则成为孤儿进程,此时init进程(id为1)会成为子进程的新的父进程;

    如果子进程先结束,则子进程会成为僵死进程!僵死进程本身并不占有CPU资源,但是它占用了进程表项,如果有很多僵死进程,那么很多正常的进程就无法注册进进程表了;因此,我们必须要对僵死进程进行回收,就用wait和waitpid;

    waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程结束。
      如果在调用 wait()时子进程已经结束,则 wait()会立即返回子进程结束状态值。
      子进程的结束状态值会由参数 status 返回,而子进程的进程识别码也会一快返回。
      如果不在意结束状态值,则参数 status 可以设成 NULL。
  
   参数 pid 为欲等待的子进程识别码,其他数值意义如下:
      pid < -1 等待进程组识别码为 pid 绝对值的任何子进程。
      pid = -1 等待任何子进程,相当于 wait()。            
      pid = 0  等待进程组识别码与目前进程相同的任何子进程。       
      pid > 0  等待任何子进程识别码为 pid 的子进程。

  参数 option 可以为 0 或下面的 OR 组合:
     WNOHANG 如果没有任何已经结束的子进程则马上返回, 不予以等待。
     WUNTRACED 如果子进程进入暂停执行情况则马上返回,但结束状态不予以理会。

  子进程的结束状态返回后存于 status,底下有几个宏可判别结束情况:
     WIFEXITED(status)如果子进程正常结束则为非 0 值。
     WEXITSTATUS(status)取得子进程 exit()返回的结束代码,一般会先用 WIFEXITED 来判断是否正常结束才能使用此宏。
     WIFSIGNALED(status)如果子进程是因为信号而结束则此宏值为真
     WTERMSIG(status) 取得子进程因信号而中止的信号代码,一般会先用 WIFSIGNALED 来判断后才使用此宏。
     WIFSTOPPED(status) 如果子进程处于暂停执行情况则此宏值为真。一般只有使用 WUNTRACED 时才会有此情况。
     WSTOPSIG(status) 取得引发子进程暂停的信号代码,一般会先用 WIFSTOPPED 来判断后才使用此宏。
     如果执行成功则返回子进程识别码(PID) ,如果有错误发生则返回返回值-1。失败原因存于 errno 中。


例子:

#include <stdio.h>
#include <unistd.h>int main()
{int i = 10;pid_t pid;int status;printf("Father's pid:%d\n", getpid());pid = fork();if(pid < 0){perror("fork failure!");return -1;}else if(pid == 0){i++;printf("Child's i = %d\n", i);sleep(7);}else{printf("Child's pis:%d\n", pid);printf("Father's i = %d\n", i);sleep(2);//    wait(&status);}return 0;
}

    

    上面的程序如果不使用wait函数对子进程进行回收,则父进程2秒正常结束后,子进程的父进程会变为init进程,可以用ps -l命令查看,达到运行的时间7秒后,子进程正常结束;如果使用了wait,则wait会使父进程等待子进程结束,子进程结束后一起退出,避免了僵死进程的产生。




这篇关于Linux系统编程_8_进程控制之fork_wait_waitpid函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/340713

相关文章

Linux流媒体服务器部署流程

《Linux流媒体服务器部署流程》文章详细介绍了流媒体服务器的部署步骤,包括更新系统、安装依赖组件、编译安装Nginx和RTMP模块、配置Nginx和FFmpeg,以及测试流媒体服务器的搭建... 目录流媒体服务器部署部署安装1.更新系统2.安装依赖组件3.解压4.编译安装(添加RTMP和openssl模块

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

在不同系统间迁移Python程序的方法与教程

《在不同系统间迁移Python程序的方法与教程》本文介绍了几种将Windows上编写的Python程序迁移到Linux服务器上的方法,包括使用虚拟环境和依赖冻结、容器化技术(如Docker)、使用An... 目录使用虚拟环境和依赖冻结1. 创建虚拟环境2. 冻结依赖使用容器化技术(如 docker)1. 创

浅析如何使用Swagger生成带权限控制的API文档

《浅析如何使用Swagger生成带权限控制的API文档》当涉及到权限控制时,如何生成既安全又详细的API文档就成了一个关键问题,所以这篇文章小编就来和大家好好聊聊如何用Swagger来生成带有... 目录准备工作配置 Swagger权限控制给 API 加上权限注解查看文档注意事项在咱们的开发工作里,API

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

CentOS系统Maven安装教程分享

《CentOS系统Maven安装教程分享》本文介绍了如何在CentOS系统中安装Maven,并提供了一个简单的实际应用案例,安装Maven需要先安装Java和设置环境变量,Maven可以自动管理项目的... 目录准备工作下载并安装Maven常见问题及解决方法实际应用案例总结Maven是一个流行的项目管理工具

Linux环境变量&&进程地址空间详解

《Linux环境变量&&进程地址空间详解》本文介绍了Linux环境变量、命令行参数、进程地址空间以及Linux内核进程调度队列的相关知识,环境变量是系统运行环境的参数,命令行参数用于传递给程序的参数,... 目录一、初步认识环境变量1.1常见的环境变量1.2环境变量的基本概念二、命令行参数2.1通过命令编程