linux select return 0

本文详细解析了 Linux 下 waitpid 函数的功能及使用方法,包括如何通过不同选项控制子进程的等待行为,并展示了具体的代码示例。

linux select return 0 

试图启动一个进程,第一次return 1, 后面都return 0 


          zoombie_pid = waitpid(-1, &child_status, WNOHANG);  

WNOHANG 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若结束,则返回该子进程的ID。持续返回值为PID,说明这个子进程已经结束。

Jun 21 22:01:31 lc4 DENG: select_ret: 0x1, num_fds : 0xc,
Jun 21 22:01:31 lc4 DENG: go event handler
Jun 21 22:01:32 lc4 DENG: select_ret: 0x0, num_fds : 0xc,
Jun 21 22:01:33 lc4 DENG: select_ret: 0x0, num_fds : 0xc,




定义函数 pid_t waitpid(pid_t pid,int * status,int options);

函数说明

编辑
如果在调用 waitpid()时子进程已经结束,则 waitpid()会立即
返回子进程结束状态值。 子进程的结束状态值会由参数 status 返回,
而子进程的进程识别码也会一起返回。如果不在意结束状态值,则
参数 status 可以设成 NULL。参数 pid 为欲等待的子进程识别码,
其他数值意义如下:
pid<-1 等待 进程组识别码为 pid 绝对值的任何子进程。
pid=-1 等待任何子进程,相当于 wait()。
pid=0 等待进程组识别码与目前进程相同的任何子进程。
pid>0 等待任何子进程识别码为 pid 的子进程。
参数options提供了一些额外的选项来控制waitpid,参数 option 可以为 0 或可以用"|"运算符把它们连接起来使用,比如:
ret=waitpid(-1,NULL,WNOHANG | WUNTRACED);
如果我们不想使用它们,也可以把options设为0,如:
ret=waitpid(-1,NULL,0);
WNOHANG 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若结束,则返回该子进程的ID。
WUNTRACED 若子进程进入暂停状态,则马上返回,但子进程的结束状态不予以理会。WIFSTOPPED(status)宏确定返回值是否对应与一个暂停子进程。
子进程的结束状态返回后存于 status,底下有几个宏可判别结束情况:
WIFEXITED(status)如果若为正常结束子进程返回的状态,则为真;对于这种情况可执行WEXITSTATUS(status),取子进程传给exit或_eixt的低8位。
WEXITSTATUS(status)取得子进程 exit()返回的结束代码,一般会先用 WIFEXITED 来判断是否正常结束才能使用此宏。
WIFSIGNALED(status)若为异常结束子进程返回的状态,则为真;对于这种情况可执行WTERMSIG(status),取使子进程结束的信号编号。
WTERMSIG(status) 取得子进程因信号而中止的信号代码,一般会先用 WIFSIGNALED 来判断后才使用此宏。
WIFSTOPPED(status) 若为当前暂停子进程返回的状态,则为真;对于这种情况可执行WSTOPSIG(status),取使子进程暂停的信号编号。
WSTOPSIG(status) 取得引发子进程暂停的信号代码,一般会先用 WIFSTOPPED 来判断后才使用此宏。
如果执行成功则返回子进程识别码(PID) ,如果有错误发生则返回
返回值-1。失败原因存于 errno 中。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/******
* waitpid.c - Simple wait usage
*********/
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>
int  main(  void  )
{
     pid_t childpid;
     int  status;
     childpid = fork();
if  ( childpid < 0 )
{
     perror "fork()"  );
     exit ( EXIT_FAILURE );
}
else  if  ( childpid == 0 )
{
     puts "In child process"  );
     sleep( 3 ); //让子进程睡眠3秒,看看父进程的行为 <--等三秒。。。。
     printf ( "\tchild pid = %d\n" , getpid());
     printf ( "\tchild ppid = %d\n" , getppid());
     exit (EXIT_SUCCESS);
}
else
{
     waitpid( childpid, &status, 0 );
     puts "in parent"  );
     printf "\tparent pid = %d\n" , getpid() );
     printf "\tparent ppid = %d\n" , getppid() );
     printf "\tchild process exited with status %d \n" , status );
}
exit (EXIT_SUCCESS);
}
1
2
3
4
5
6
7
8
9
10
[root@localhost src] # gcc waitpid.c
[root@localhost src] # ./a.out
In child process
child pid = 4469
child ppid = 4468
in  parent
parent pid = 4468
parent ppid = 4379
child process exited with status 0
[root@localhost src] #
如果将上面“waitpid( childpid, &status, 0 );”行注释掉,程序执行效果如下:
1
2
3
4
5
6
7
8
[root@localhost src] # ./a.out
In child process
in  parent
parent pid = 4481
parent ppid = 4379
child process exited with status1
[root@localhost src] # child pid = 4482
child ppid = 1
子进程还没有退出, 父进程已经退出了。

### Linux C语言中 `select` 函数返回0(超时)的原因及解决方法 在Linux环境下,C语言中的 `select` 函数用于监控多个文件描述符的状态变化。当 `select` 函数返回值为0时,表示函数执行过程中没有检测到任何文件描述符变为就绪状态,并且超时时间已到达[^1]。 #### 返回0的常见原因 1. **超时时间设置过短** 如果 `timeout` 参数被设置为一个非常短的时间间隔(例如几毫秒),而在此时间内没有文件描述符变为就绪状态,则 `select` 会直接返回0,表示超时[^3]。 2. **文件描述符未准备好** 如果在指定的超时时间内,所有传入的文件描述符集合(`readfds`、`writefds` 和 `exceptfds`)均未发生预期的事件(如可读、可写或异常条件),则 `select` 也会返回0。 3. **错误的文件描述符集合配置** 如果传递给 `select` 的文件描述符集合为空,或者这些描述符实际上并未被正确初始化,则可能导致 `select` 在超时后返回0,因为没有任何有效的文件描述符可以被监控[^2]。 #### 解决方案 1. **调整超时时间** 根据实际需求合理设置 `timeout` 参数。如果需要更长时间等待文件描述符变为就绪状态,可以将 `timeout` 设置为更大的值,甚至设置为 `NULL` 以实现无限期等待(需注意可能带来的阻塞问题)[^1]。 2. **检查文件描述符集合** 确保传递给 `select` 的文件描述符集合已被正确初始化,并包含至少一个有效的文件描述符。可以通过调试工具或日志记录来验证集合的内容是否符合预期[^2]。 3. **优化轮询机制** 如果频繁出现超时情况,考虑使用更高效的IO多路复用技术(如 `poll` 或 `epoll`),这些技术可能更适合处理大量文件描述符或需要更高性能的应用场景[^3]。 4. **代码示例** 下面是一个简单的 `select` 使用示例,展示如何正确设置超时时间和文件描述符集合: ```c #include <stdio.h> #include <sys/select.h> #include <sys/time.h> #include <unistd.h> int main() { fd_set readfds; struct timeval timeout; int fd = STDIN_FILENO; // 监控标准输入 FD_ZERO(&readfds); FD_SET(fd, &readfds); timeout.tv_sec = 5; // 设置超时时间为5秒 timeout.tv_usec = 0; int ret = select(fd + 1, &readfds, NULL, NULL, &timeout); if (ret == -1) { perror("select error"); } else if (ret == 0) { printf("Timeout occurred! No data after 5 seconds.\n"); } else { if (FD_ISSET(fd, &readfds)) { printf("Data is available now.\n"); // 处理数据 } } return 0; } ``` #### 注意事项 - 在实际应用中,应根据具体业务需求调整超时时间,避免因过短或过长的超时设置影响程序性能或用户体验[^3]。 - 当 `select` 返回0时,无需调用 `errno`,因为这并不表示发生错误,而是正常的功能行为。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值