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 rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

基于Python实现自动化邮件发送系统的完整指南

《基于Python实现自动化邮件发送系统的完整指南》在现代软件开发和自动化流程中,邮件通知是一个常见且实用的功能,无论是用于发送报告、告警信息还是用户提醒,通过Python实现自动化的邮件发送功能都能... 目录一、前言:二、项目概述三、配置文件 `.env` 解析四、代码结构解析1. 导入模块2. 加载环

linux系统上安装JDK8全过程

《linux系统上安装JDK8全过程》文章介绍安装JDK的必要性及Linux下JDK8的安装步骤,包括卸载旧版本、下载解压、配置环境变量等,强调开发需JDK,运行可选JRE,现JDK已集成JRE... 目录为什么要安装jdk?1.查看linux系统是否有自带的jdk:2.下载jdk压缩包2.解压3.配置环境

Linux搭建ftp服务器的步骤

《Linux搭建ftp服务器的步骤》本文给大家分享Linux搭建ftp服务器的步骤,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录ftp搭建1:下载vsftpd工具2:下载客户端工具3:进入配置文件目录vsftpd.conf配置文件4: