Linux的wait函数详解

本文深入探讨了进程管理中的核心函数wait与waitpid的工作原理,详细解释了它们如何用于监控与等待子进程的结束,以及如何通过这些函数获取子进程的结束状态和PID。通过两个示例程序演示了函数的具体应用。
【转自:http://hi.baidu.com/_hstyle/blog/item/fb7b12c4447390af8226acc5.html】

    进程一旦调用了 wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程,wait 就会收集这个子进程的信息, 并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。

wait(等待子进程中断或结束)
相关函数 waitpid,fork
表头文件
#include<sys/types.h>
#include<sys/wait.h>
定义函数 pid_t wait (int * status);
函数说明
wait()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用wait()时子进程已经结束,则wait()会立即返回子进程结束状态值。子进程的结束状态值会由参数status 返回,而子进程的进程识别码也会一快返回。如果不在意结束状态值,则参数status 可以设成NULL。子进程的结束状态值请参考waitpid()。
返回值
如果执行成功则返回子进程识别码(PID),如果有错误发生则返回-1。失败原因存于errno 中。


范例一
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>
int main()
{
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);
}
}

执行
This is the child process.pid=1501
This is the parent process .wait for child...
child’s pid =1501,exit status =5


范例二

#include<iostream>
#include<unistd.h>
#include<sys/wait.h>
using namespace std;
int main(void)
{
pid_t pid;
pid =fork();
if (pid<0)
exit(0);
else if (pid == 0)
{
//如果是子进程 睡眠20秒
cout<<"children : "<<getpid()<<endl;
sleep(20);
}
else
{ cout<<"hello! i'm parent process!"<<endl;
//如果是父进程在这里等待
pid_t pr = wait(NULL);
cout<<pr<<endl;
}
return 0;
}

waitpid(等待子进程中断或结束)
相关函数 wait,fork
表头文件
#include<sys/types.h>
#include<sys/wait.h>
定义函数 pid_t waitpid(pid_t pid,int * status,int options);
函数说明
waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用waitpid()时子进程已经结束,则waitpid()会立即返回子进程结束状态值。子进程的结束状态值会由参数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 中。

范例参考wait()。

`poll_wait` 函数Linux 内核中用于实现轮询等待的一个重要函数,其主要作用是将当前进程挂起,等待某个特定的事件发生,例如数据从 IO 设备上到达或者设备错误等。当事件发生时,`poll_wait` 将会唤醒等待的进程,使其可以继续执行。 下面详细介绍一下 `poll_wait` 函数的使用方法。 ## 语法 ```c void poll_wait(struct file *filp, wait_queue_head_t *queue, struct wait_queue_entry *wait); ``` ## 参数 - `filp`:待等待的文件指针; - `queue`:等待队列的头指针; - `wait`:等待队列的条目指针。 ## 功能 将当前进程加入到等待队列中,等待某个特定的事件发生。 ## 注意事项 1. `poll_wait` 函数只能在进程的上下文中使用,不能在中断上下文中使用。 2. 在调用 `poll_wait` 函数之前,必须先调用 `init_waitqueue_head` 函数对等待队列进行初始化。 3. 在等待队列中,每个等待事件都必须使用唯一的等待队列条目。 ## 示例 下面是一个示例程序,演示了如何使用 `poll_wait` 函数: ```c DEFINE_WAIT(wait); init_waitqueue_head(&wait_queue_head); // 等待事件发生 poll_wait(file, &wait_queue_head, &wait); // 判断事件是否已经发生 if (event_occurred) { // 唤醒等待队列中的进程 wake_up_interruptible(&wait_queue_head); } ``` 在上面的示例程序中,首先使用 `DEFINE_WAIT` 宏定义了一个等待队列条目,然后使用 `init_waitqueue_head` 函数对等待队列进行初始化。接着,使用 `poll_wait` 函数将当前进程加入到等待队列中,等待事件发生。当事件发生时,使用 `wake_up_interruptible` 函数唤醒等待队列中的进程,使其可以继续执行。 总之,`poll_wait` 函数Linux 内核中实现轮询等待的一个重要函数,它可以方便地实现进程挂起和等待事件发生的功能,是 Linux 内核中非常常用的一个函数
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值