1. 函数说明:
kill和raise是用来发送信号的:
kill把信号发送给进程或进程组;
raise把信号发送给(进程)自身.
他们的原型如下:
#include <signal.h>
int kill(pid_t pid, int signo);
int raise(int signo);
成功则返回0, 出错则返回-1
int kill(pid_t pid, int signo);
int raise(int signo);
成功则返回0, 出错则返回-1
从原型上可以看出, raise函数是可以通过kill实现的.
raise(signo);
等价于:
kill(getpid(), signo);
2. pid参数:
kill函数中的pid参数, 它有以下4种情况:
- pid > 0: 将该信号发送给进程ID为pid的进程.
- pid == 0: 将该信号发送给与发送进程属于同一进程组的所有进程(不包括内核进程和init进程). 此时, 发送进程必须具有向这些进程发送信号的权限.
- pid < 0: 将该信号发给其进程组ID等于pid绝对值的所有进程(不包括内核进程和init进程). 此时, 发送进程必须具有向这些进程发送信号的权限.
- pid == -1: 将该信号发送给发送进程有权限向它们发送信号的系统上的所有进程.(不包括内核进程和init进程).
3. signo参数:
POSIX.1将编号为0的信号定义为空信号. 如果signo参数是0, 则kill仍执行正常的错误检查, 但不发送信号. 这被用来确定一个进程是否存在.
4. 例子说明
返回值说明: 成功执行时,返回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
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
在确信fork调用成功后,子进程睡眠100秒,然后退出。
同时父进程在子进程上调用waitpid函数,但使用了WNOHANG选项(WNOHANG如果没有任何已经结束的子进程则马上返回,不予以等待),
所以调用waitpid后立即返回。父进程接着杀死子进程,如果kill执行失败,
返回-1,否这返回0。如果kill执行失败,父进程第二次调用waitpid,
保证他在子进程退出后再停止执行。否则父进程显示一条成功消息后退出。