天天看點

信号signal

一、信号

         信号用于處理異步事件,信号的通信方式了解起來還是有一定難度的。它既可以在一個程序内通信,發送信号給程序,又可以用于不同程序的通信。

         信号在驅動中的應用比較廣泛,在應用中用到的多半是一些linux指令操作。

二、函數說明

1、alarm

鬧鐘函數

alarm也稱為鬧鐘函數,它可以在程序中設定一個定時器,當定時器指定的時間到時,它向程序發送SIGALRM信号。可以設定忽略或者不捕獲此信号,如果采用預設方式其動作是終止調用該alarm函數的程序。

2、函數名: signal

表頭檔案#include<signal.h>

功 能:設定某一信号的對應動作

函數原型:void (*signal(int signum,void(* handler)(int)))(int);

或者:typedef void (*sig_t)( int );

sig_t signal(int signum,sig_t handler);

3、頭檔案

#include <signal.h>

sigemptyset(sigset_t *set)初始化由set指定的信号集,信号集裡面的所有信号被清空;

sigfillset(sigset_t *set)調用該函數後,set指向的信号集中将包含linux支援的62種信号;

sigaddset(sigset_t *set, int signum)在set指向的信号集中加入signum信号;

sigdelset(sigset_t *set, int signum)在set指向的信号集中删除signum信号;

sigismember(const sigset_t *set, int signum)判定信号signum是否在set指向的信号集中。

int sigaction( int sig, const struct sigaction *act,struct sigaction *oact )檢查、修改和指定信号相關聯的信号響應。

4、參數結構sigaction定義如下

struct sigaction {

void (*sa_handler)(int);

void (*sa_sigaction)(int, siginfo_t *, void *);

sigset_t sa_mask;

int sa_flags;

void (*sa_restorer)(void);

};

信号處理函數可以采用void (*sa_handler)(int)或void (*sa_sigaction)(int, siginfo_t *, void *)。到底采用哪個要看sa_flags中是否設定了SA_SIGINFO位,如果設定了就采用void (*sa_sigaction)(int, siginfo_t *, void *),此時可以向處理函數發送附加資訊;預設情況下采用void (*sa_handler)(int),此時隻能向處理函數發送信号的數值。

sa_handler此參數和signal()的參數handler相同,代表新的信号處理函數,其他意義請參考signal()。

sa_mask 用來設定在處理該信号時暫時将sa_mask 指定的信号集擱置。

sa_restorer 此參數沒有使用。

sa_flags 用來設定信号處理的其他相關操作,下列的數值可用。

sa_flags還可以設定其他标志:

SA_RESETHAND:當調用信号處理函數時,将信号的處理函數重置為預設值SIG_DFL

SA_RESTART:如果信号中斷了程序的某個系統調用,則系統自動啟動該系統調用

SA_NODEFER :一般情況下, 當信号處理函數運作時,核心将阻塞該給定信号。但是如果設定了 SA_NODEFER标記, 那麼在該信号處理函數運作時,核心将不會阻塞該信号

三、程式

示例一:

#include<unistd.h>
#include<stdio.h>
#include<signal.h>

void handler()
{
	printf("hello\n");
}

int main(void)
{
	int i;
	signal(SIGALRM,	handler);
	alarm(5);
	
	for(i=1;i<7;i++){
		printf("sleep %d....\n",i);
		sleep(1);
	}
	
	return 0;
}
           

示例二:

#include <unistd.h>  
#include <signal.h>  
#include <sys/types.h>  
#include <stdlib.h>  
#include <stdio.h>  

void handler(int sig)  
{  
    printf("Handler the signal %d\n", sig);  
}  
  
int main(void)  
{ 
	/*信号集*/
    sigset_t sigset;//用于記錄屏蔽字  
    struct sigaction act; 
	
    //清空信号集  
    sigemptyset(&sigset);  //初始化信号集,把這個信号集清空
    sigemptyset(&ign);  
    //向信号集中添加信号SIGINT ,信号是鍵盤的ctrl+c
    sigaddset(&sigset, SIGINT);  
  
    //設定處理函數和信号集      
    act.sa_handler = handler;  
    sigemptyset(&act.sa_mask);  
    act.sa_flags = 0;  
    sigaction(SIGINT, &act, 0);  
  
    printf("Wait the signal SIGINT...\n");  
    pause();//挂起程序,等待信号  
  
    //設定程序屏蔽字,在本例中為屏蔽SIGINT   
    sigprocmask(SIG_SETMASK, &sigset, 0);     
    printf("Please press Ctrl+c in 10 seconds...\n");  
    sleep(10); 
	

    //在信号集中删除信号SIGINT  
    sigdelset(&sigset, SIGINT);  
    printf("Wait the signal SIGINT...\n"); 
	
    //将程序的屏蔽字重新設定,即取消對SIGINT的屏蔽  
    //并挂起程序  
    sigsuspend(&sigset);  
  
    printf("The app will exit in 5 seconds!\n");  
    sleep(5);  
    exit(0);  
}