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

相关文章

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

linux-基础知识3

打包和压缩 zip 安装zip软件包 yum -y install zip unzip 压缩打包命令: zip -q -r -d -u 压缩包文件名 目录和文件名列表 -q:不显示命令执行过程-r:递归处理,打包各级子目录和文件-u:把文件增加/替换到压缩包中-d:从压缩包中删除指定的文件 解压:unzip 压缩包名 打包文件 把压缩包从服务器下载到本地 把压缩包上传到服务器(zip

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor