信号通信

信号通信

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

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

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

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

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

常见信号:

下面是几种常见的信号:

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 信号处理函数名:使用该函数处理

信号实例:

/**********************************************************
*实验要求:   用signal注册信号处理函数,并用kill发送相应信号触发信
*           号处理。
*功能描述:   在进程中为SIGBUS注册处理函数,并向该进程发送SIGBUS信
*           号来触发已注册的处理函数。
*日    期:   2010-9-17
*作    者:   国嵌
**********************************************************/
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>

/*
	* 自定义的信号处理函数
	* */
void my_func(int sign_no)
{
	if(sign_no==SIGBUS)
	{
		printf("I have get SIGBUS\n");
	}
}

/*
 * 程序入口
 * */
int main()
{
	printf("Waiting for signal SIGBUS \n ");
	
	/*注册信号处理函数*/
	signal(SIGBUS,my_func);
	
	pause();//将进程挂起直到捕捉到信号为止
	exit(0);
}


程序运行步骤如下:

首先启动进程

[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下
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值