linux 进程等待 wait 、 waitpid

waitpid() 与 wait() 功能相似,都是用户主进程等待子进程结束或中断. 可用于进程之间的同步

wait 函数原型

pid_t wait(int *status);

函数说明

wait() 会临时停止眼下进程的运行,直到有信号来到或子进程结束.假设在调用wait() 时子进程已经结束,则 wait() 会立即返回子进程结束状态值.子进程的结束状态值会由參数 status 返回,而子进程的进程识别码也会一块返回.假设不在意结束状态值,则參数ststus能够设为 NULL.子进程的结束状态值请參考以下的waitpid().

返回值

假设运行成功则返回子进程识别码(PID), 假设有发生错误则返回 -1, 失败原因存于 errno 中.

演示样例代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>

int main(int argc, char *argv[])
{
    pid_t pid;
    int status, i;
    if(fork() == 0) {
        printf("This is the child process pid = %d\n",getpid());
        exit(5);
    } else {
        sleep(1);
        printf("This is the parent process , wait for child...\n");
        pid = wait(&status);
        i = WEXITSTATUS(status);
        printf("child‘s pid = %d. exit status = %d\n", pid, i);
    }
    return 0;
}

waitpid 函数原型

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

函数说明

waitpid() 会临时停止眼下进程的运行,直到有信号来到或子进程结束. 假设在调用 waitpid() 时子进程已经结束,则 waitpid() 会立即返回子进程结束状态值. 子进程的结束状态值会由參数 status 返回,而子进程的进程识别码也会一块返回.假设不在意结束状态值,则參数ststus能够设为NULL.參数pid为欲等待的子进程识别码.其数值意义例如以下:

pid > 0 时,仅仅等待进程id等于pid的子进程,无论其他已经有多少子进程运行结束退出,仅仅要指定的子进程还没有结束,waitpid就会一直等下去.

pid = -1 时,等待不论什么一个子进程退出,没有不论什么限制,此时 waitpid 和 wait 的作用一模一样.

pid = 0 时,等待统一进程组中的不论什么子进程,假设子进程已经增加了别的进程组,waitpid 不会对它做不论什么理睬.

pid < -1 时, 等待一个指定进程组中的不论什么子进程,这个进程组的ID等于pid的绝对值。

參数 options 的值有以下几种类型:

WNOHANG 假设没有不论什么已经结束的子进程则立即返回, 不予以等待。  

WUNTRACED 假设子进程进入暂停运行情况则立即返回,但结束状态不予以理会。

假设不用以上两个宏。还能够用 0 作为第三个參数传入。

注: wait() 函数就是经过包装的 waitpid(),查看 <内核源代码文件夹>/include/unistd.h 文件 就能够看到例如以下程序段

static inline pid_t wait(int *wait_stat)
{
    return waitpid(-1,wait_stat,0);
}

返回值

当正常返回的时候 waitpid 返回收集到的子进程的ID;

假设设置了 WNOHANG, 而调用中waitpid 发现没有已退出的子进程可收集,则返回0;

假设调用中出错,则返回-1,并重置errno的值。

子进程的结束状返回后存于 status,地下有几个宏可判别结束情况

WNOHANG 假设没有不论什么已经结束的子进程则立即返回, 不予以等待。

  

WUNTRACED 假设子进程进入暂停运行情况则立即返回,但结束状态不予以理会。

  

子进程的结束状态返回后存于 status,底下有几个宏可判别结束情况:  

WIFEXITED(status)假设子进程正常结束则为非 0 值。  

WEXITSTATUS(status)取得子进程 exit()返回的结束代码,通常会先用

WIFEXITED 来推断是否正常结束才干使用此宏。  

WIFSIGNALED(status)假设子进程是由于信号而结束则此宏值为真  

WTERMSIG(status) 取得子进程因信号而中止的信号代码,通常会先用 WIFSIGNALED 来推断后才使用此宏。  

WIFSTOPPED(status) 假设子进程处于暂停运行情况则此宏值为真。

一般仅仅有使用 WUNTRACED 时才会有此情况。  

WSTOPSIG(status) 取得引发子进程暂停的信号代码,通常会先用 WIFSTOPPED 来推断后才使用此宏。

演示样例代码:

#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
main()
{
    pid_t pc, pr;
    pc=fork();
    if(pc<0)
        printf("Error occured on forking.\n");
    else if(pc==0) {
        sleep(10);
        exit(0);
    }
    do {
        pr=waitpid(pc, NULL, WNOHANG);
        if(pr==0) {
            printf("No child exited\n");
            sleep(1);
        }
    } while(pr==0);
    if(pr==pc)
        printf("successfully get child %d\n", pr);
    else
        printf("some error occured\n");

}
时间: 2024-10-06 16:12:42

linux 进程等待 wait 、 waitpid的相关文章

转载:进程退出状态--waitpid status意义

最近遇到一个进程突然退出的问题,由于没有注册signalhandler所以没有捕捉到任何信号. 但是从log中看到init waitpid返回的status为0x008b,以前对status不是很了解,下面的文章对status有比较全面的介绍. 转至http://tsecer.blog.163.com/blog/static/15018172012323975152/ 一.和子进程同步在linux系统中,父进程通常需要通过waitpid来等待/获取子进程状态变化情况,而这个主要就是通过waitX

六、Linux进程控制

1. Linux进程概述 进程是一个程序一次执行的过程,它和程序有本质区别. 程序是静态的,它是一些保存在磁盘上的指令的有序集合:而进程是一个动态的概念,它是一个运行着的程序,包含了进程的动态创建.调度和消亡的过程,是Linux的基本调度单位. 那么从系统的角度看如何描述并表示它的变化呢?在这里,是通过进程控制块(PCB)来描述的.进程控制块包含了进程的描述信息.控制信息以及资源信息,它是进程的一个静态描述. 内核使用进程来控制对CPU和其他系统资源的访问,并且使用进程来决定在CPU上运行哪个程

Linux进程间的通信

一.管道 管道是Linux支持的最初Unix IPC形式之一,具有以下特点: A. 管道是半双工的,数据只能向一个方向流动: B. 需要双工通信时,需要建立起两个管道: C. 只能用于父子进程或者兄弟进程之间(具有亲缘关系的进程): D. 单独构成一种独立的文件系统:管道对于管道两端的进程而言,就是一个文件,但它不是普通的文件,它不属于某种文件系统,而是自立门户,单独构成一种文件系统,并且只存在与内存中. 匿名管道的创建:该函数创建的管道的两端处于一个进程中间,在实际应用中没有太大意义;因此,一

Linux进程管理知识整理

Linux进程管理知识整理 1.进程有哪些状态?什么是进程的可中断等待状态?进程退出后为什么要等待调度器删除其task_struct结构?进程的退出状态有哪些? TASK_RUNNING(可运行状态) TASK_INTERRUPTIBLE(可中断等待状态) TASK_UNINTERRUPTIBLE(不可中断等待状态) TASK_STOPPED(进程被其它进程设置为暂停状态) TASK_TRACED(进程被调试器设置为暂停状态) TASK_DEAD(退出状态) 进程由于所需资源得不到满足,从而进入

Linux进程控制知识总结

目录 一:进程标识符(ID) 二:进程操作 2.1创建一个进程 2.2 fork函数出错情况 2.3创建一个共享空间的子进程 2.4退出程序 2.5设置进程所有者 三:执行程序 3.1 exec函数 3.2 执行解释器文件 3.3在程序中执行Shell命令 四:关系操作符 4.1等待进程退出 4.2 等待指定的进程 进程控制 -- 一步 一:进程标识符(ID) 进程ID是用来标识进程的编号,就像身份证一样.不同的进程有不同的ID,可以通过ID来查询进程.进程标识符的类型是pit_t,其本质是一个

Linux进程管理——fork()和写时复制

写时复制技术最初产生于Unix系统,用于实现一种傻瓜式的进程创建:当发出fork(  )系统调用时,内核原样复制父进程的整个地址空间并把复制的那一份分配给子进程.这种行为是非常耗时的,因为它需要: ·      为子进程的页表分配页面 ·      为子进程的页分配页面 ·      初始化子进程的页表 ·      把父进程的页复制到子进程相应的页中 创建一个地址空间的这种方法涉及许多内存访问,消耗许多CPU周期,并且完全破坏了高速缓存中的内容.在大多数情况下,这样做常常是毫无意义的,因为许多

Linux 进程间通讯方式 pipe()函数 (转载)

转自:http://blog.csdn.net/ta893115871/article/details/7478779 Linux 进程间通讯方式有以下几种: 1->管道(pipe)和有名管道(fifo). 2->消息队列 3->共享内存 4->信号量 5->信号(signal) 6->套接字(sicket) 在这里我们看一下第一种====管道(pipe).有名管道(fifo)见其它文章. eg :我们以前学的命令 cat  file | grep  "abc

linux 进程(二) --- 进程的创建及相关api

一.进程的创建fork()函数 由fork创建的新进程被称为子进程(child process).该函数被调用一次,但返回两次.两次返回的区别是子进程的返回值是0,而父进程的返回值则是 新子进程的进程ID.将子进程ID返回给父进程的理由是:因为一个进程的子进程可以多于一个,所有没有一个函数使一个进程可以获得其所有子进程的进程ID.fork使子进程得到返回值0的理由是:一个进程只会有一个父进程,所以子进程总是可以调用getppid以获得其父进程的进程ID(进程 ID  0总是由交换进程使用,所以一

Linux - 进程 (二) 进程创建

详细见:https://github.com/ZhangzheBJUT/linux 一 进程概述 一个进程都由另一个称之为父进程的进程启动,被父进程启动的进程叫做子进程.Linux系统启动时候,它将运行一个名为init的进程,该进程是系统运行的第一个进程,它的进程号为1,它负责管理其它进程,可以把它看做是操作系统进程管理器,它是其它所有进程的祖先进程.系统中的进程要么是由init进程启动,要么是由init进程启动的其他进程启动. 使用ps命令输出中的PPID栏给出的是父进程的进程ID,它是启动这