信号通信--经典

信号通信

    信号(singal)机制是Unix系统中最为古老的进程间通信机制,很多条件能够产生一个信号:

    1  当用户按下某一按键时,产生信号。

    2 硬件异常产生信号:除数为0、无效的存储访问等等。这些情况往往由硬件检测到,将其通知内核,然后内核产生适当的信号通知进程,例如,内核对正访问一个无效存储区的进程产生一个SIGSEGV信号。

    3 进程可以使用Kill函数将信号发送给另外一个信号。

    4 进程可用Kill命令将信号发送给其他进程。

常见信号:

信号通信--经典_第1张图片

下面是几种常见的信号:

SIGHUP:从终端发出的结束信号

SIGINT: 来自键盘的中断信号(ctrl+c)

SIGKILL:该信号结束接收信号的进程

SIGTERM:kill命令发出的信号

SIGCHLD:标识子进程停止或结束的信号

SIGSTOP:来自键盘(ctrl+z)或调试程序的停止执行信号

信号处理

当信号出现时,将按照三种方式中的一种处理信号:

1 忽略此信号

    大多数信号都是按照这种方式进行处理的,当有两种信号却决不能被忽略。SIGKILL和SIGSTOP。这两种信号不能够被忽略的原因是:它们向超级用户提供一种终止或停止进程的方法。

2 执行用户希望的动作

    通知内核在某种信号发生时,调用一个用户函数。在用户函数中,执行用户希望的处理。

3 执行系统默认动作

   对大多数信号系统默认动作是终止该进程。

信号发送

    发送信号的主要函数有kill和raise。

    区别在于:kill既可以向自身发送信号,也可以向其它进程发送信号。与kill函数不同的是,raise是向进程自身发送信号。

#include<sys/types.h>

#include<signal.h>

int kill(pid_t pid,int signo)

int raise(int signo)

Kill的pid参数有四种不同情况:

1 pid>0 :将信号发送给进程ID为pid的进程。

2 pid==0 :将信号发送给同组的进程。

3 pid<0 :将信号发送给其它进程组ID为pid绝对值的进程。

4 pid==-1 :将信号发送给所有进程。

Alarm发送信号

    使用alarm函数可以设置一个时间值(闹钟时间),当所设置的时间到了时,产生SIGALRM信号(发送给进程本身)。如果不捕捉此信号,则默认动作是终止改进程。

    #include<unistd.h>

    unsigned int alarm(unsigned int seconds)

    seconds:经过指定的seconds秒后会产生信号SIGALRM。

    每个进程只能有一个闹钟时间。如果在调用alarm时,之前已经为该进程设置过闹钟时间,而且他还没有超时,以前登记的闹钟时间则被新的闹钟时间替代。

    如果有以前登记尚未超时的闹钟时间,而这次seconds值为0,则表示取消以前的闹钟。

pause函数

    pause函数是调用进程挂起知道捕捉到一个信号。

    #include<unistd.h>

   int pause(void)

    只有执行一个信号处理函数后,挂起才结束。

信号处理

    当信号捕捉到某个信号时,可以忽略该信号或是使用指定的函数来处理该信号,或者使用系统默认方式。

    信号处理的主要方法有两种,一种是使用简单的signal函数,另一种是使用信号集函数组。

signal函数

#include<signal.h>

void ( *signal(int signo,void(*func)(int)))(int)

如何理解?

typedef void(*sighandler_t)(int)

sighandler_t signal(int signum,sighandler_t handler)

Func可能的值是:

1 SIG_IGN:忽略此信号

2 SIG_DFL:按系统默认方式处理

3 信号处理函数名:使用该函数处理

信号实例:

[cpp]  view plain copy
  1. /********************************************************** 
  2. *实验要求:   用signal注册信号处理函数,并用kill发送相应信号触发信 
  3. *           号处理。 
  4. *功能描述:   在进程中为SIGBUS注册处理函数,并向该进程发送SIGBUS信 
  5. *           号来触发已注册的处理函数。 
  6. *日    期:   2010-9-17 
  7. *作    者:   国嵌 
  8. **********************************************************/  
  9. #include <signal.h>  
  10. #include <stdio.h>  
  11. #include <stdlib.h>  
  12.   
  13. /* 
  14.     * 自定义的信号处理函数 
  15.     * */  
  16. void my_func(int sign_no)  
  17. {  
  18.     if(sign_no==SIGBUS)  
  19.     {  
  20.         printf("I have get SIGBUS\n");  
  21.     }  
  22. }  
  23.   
  24. /* 
  25.  * 程序入口 
  26.  * */  
  27. int main()  
  28. {  
  29.     printf("Waiting for signal SIGBUS \n ");  
  30.       
  31.     /*注册信号处理函数*/  
  32.     signal(SIGBUS,my_func);  
  33.       
  34.     pause();//将进程挂起直到捕捉到信号为止  
  35.     exit(0);  
  36. }  


程序运行步骤如下:

首先启动进程

[root@localhost 4]# ./mysignal

Waiting for signal SIGBUS

打开另外一个终端,要发送信号,首先要查看此程序的进程号,使用ps -aux命令查看

在新终端输入如下命令发送信号

[root@localhost 4]# kill -s SIGBUS 3687

终端1的结果如下:

[root@localhost 4]# ./mysignal

Waiting for signal SIGBUS

I have get SIGBUS

以上为程序运行的步骤与结果,蓝色表示为在终端1下,绿色表示在终端2下

你可能感兴趣的:(信号通信--经典)