kill waitpid与wait函数的使用

wait的函数原型是:  

#include<sys/types.h>

#include <sys/wait.h>

pid_t wait(int *status)     

      进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程, wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。     参数status用来保存被收集进程退出时的一些状态,它是一个指向int类型的指针。但如果我们对这个子进程是如何死掉的毫不在意,只想把这个僵尸进程消灭掉,(事实上绝大多数

情况下,我们都会这样想),我们就可以设定这个参数为NULL,就象下面这样:     

pid = wait(NULL);

如果成功,wait会返回被收集的子进程的进程ID,如果调用进程没有子进程,调用就会失败,此时wait返回-1,同时errno被置为ECHILD。    

 

waitpid系统调用在Linux函数库中的原型是:   

#include <sys/types.h>#include <sys/wait.h>

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

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

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

● 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;当pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD

 

● pid  从参数的名字pid和类型pid_t中就可以看出,这里需要的是一个进程ID。但当pid取不同的值时,在这里有不同的意义。

pid>0时,只等待进程ID等于pid的子进程,不管其它已经有多少子进程运行结束退出了,只要指定的子进程还没有结束,waitpid就会一直等下去。     

pid=-1时,等待任何一个子进程退出,没有任何限制,此时waitpid和wait的作用一模一样。     

pid=0时,等待同一个进程组中的任何子进程,如果子进程已经加入了别的进程组,waitpid不会对它做任何理睬。     

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

● options  options提供了一些额外的选项来控制waitpid,目前在Linux中只支持WNOHANG和WUNTRACED两个选项,这是两个常数,可以用"|"运算符把它们连接起来使用,比如:

  ret=waitpid(-1,NULL,WNOHANG | WUNTRACED);

如果我们不想使用它们,也可以把options设为0,如:   

ret=waitpid(-1,NULL,0);     

如果使用了WNOHANG参数调用waitpid,即使没有子进程退出,它也会立即返回,不会像wait那样永远等下去。

waitpid的返回值比wait稍微复杂一些,一共有3种情况:  

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

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

● 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;当pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD;

这两个

函数返回的子进程状态都保存在status指针中, 用以下3个宏可以检查该状态:

WIFEXITED(status): 若为正常终止, 则为真. 此时可执行 WEXITSTATUS(status): 取子进程传送给exit或_exit.

WIFSIGNALED(status): 若为异常终止, 则为真. 此时可执行 WTERMSIG(status): 取使子进程终止的信号编号.

WIFSTOPPED(status): 若为当前暂停子进程, 则为真. 此时可执行 WSTOPSIG(status): 取使子进程暂停的信号编号

 

KILL功能描述:

用于向任何进程组或进程发送信号。

1 #include <sys/types.h>
2 
3 #include <signal.h>
4 
5 int kill(pid_t pid, int sig);
6 
7 


参数: 
pid:可能选择有以下四种

1. pid大于零时,pid是信号欲送往的进程的标识。
2. pid等于零时,信号将送往所有与调用kill()的那个进程属同一个使用组的进程。
3. pid等于-1时,信号将送往所有调用进程有权给其发送信号的进程,除了进程1(init)。
4. pid小于-1时,信号将送往以-pid为组标识的进程。

sig:准备发送的信号代码,假如其值为零则没有任何信号送出,但是系统会执行错误检查,通常会利用sig值为零来检验某个进程是否仍在执行。

返回值说明:

成功执行时,返回0。

失败返回-1,errno被设为以下的某个值

EINVAL:指定的信号码无效(参数 sig 不合法)

EPERM:权限不够无法传送信号给指定进程

ESRCH:参数 pid 所指定的进程或进程组不存在

代码
 1 #include <sys/wait.h>
 2 #include <sys/types.h>
 3 #include <stdio.h>
 4 #include <stdlib.h>
 5 #include <signal.h>
 6 
 7 int main( void )
 8 {
 9     pid_t childpid;
10     int status;
11     int retval;
12     
13     childpid = fork();
14     if ( -1 == childpid )
15     {
16         perror( "fork()" );
17         exit( EXIT_FAILURE );
18     }
19     else if ( 0 == childpid )
20     {
21         puts( "In child process" );
22         sleep( 100 );//让子进程睡眠,看看父进程的行为
23         exit(EXIT_SUCCESS);
24     }
25     else
26     {
27         if ( 0 == (waitpid( childpid, &status, WNOHANG )))
28         {
29             retval = kill( childpid,SIGKILL );
30             
31             if ( retval )
32             {
33                 puts( "kill failed." );
34                 perror( "kill" );
35                 waitpid( childpid, &status, 0 );
36             }
37             else
38             {
39                 printf( "%d killed\n", childpid );
40             }
41             
42         }
43     }
44     
45     exit(EXIT_SUCCESS);
46 }
47 //-----------------
48 [root@localhost src]# gcc killer.c
49 [root@localhost src]# ./a.out
50 In child process
51 4545 killed

 

 

 

发布了8 篇原创文章 · 获赞 4 · 访问量 2万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览