亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? linux_c_sig.txt

?? 基于LINUX操作系統下的各種詳細配置(如FTP
?? TXT
字號:
Linux下C語言編程--信號處理函數- -
Tag: Linux    C語言    編程    信號    處理函數                                           
前言:這一章我們討論一下Linux下的信號處理函數. 
     Linux下的信號處理函數: 
1.信號的產生 
2.信號的處理 
3.其它信號函數  

--------------------------------------------------------------------------------
一個實例 
1。信號的產生 
    Linux下的信號可以類比于DOS下的INT或者是Windows下的事件.在有一個信號發生時候相信的信號就會發送給相應的進程.在Linux下的信號有以下幾個. 我們使用 kill -l 命令可以得到以下的輸出結果: 

1) SIGHUP  2) SIGINT  3) SIGQUIT  4) SIGILL
5) SIGTRAP  6) SIGABRT  7) SIGBUS  8) SIGFPE
9) SIGKILL 10) SIGUSR1 11) SIGSEGV 12) SIGUSR2
13) SIGPIPE 14) SIGALRM 15) SIGTERM 17) SIGCHLD
18) SIGCONT 19) SIGSTOP 20) SIGTSTP 21) SIGTTIN
22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO
30) SIGPWR

關于這些信號的詳細解釋請查看man 7 signal的輸出結果. 信號事件的發生有兩個來源:一個是硬件的原因(比如我們按下了鍵盤),一個是軟件的原因(比如我們使用系統函數或者是命令發出信號). 最常用的四個發出信號的系統函數是kill, raise, alarm和setitimer函數. setitimer函數我們在計時器的使用 那一章再學習. 
#include  
#include  
        #include  

int kill(pid_t pid,int sig);
int raise(int sig);
unisigned int  alarm(unsigned int seconds);

kill系統調用負責向進程發送信號sig.
如果pid是正數,那么向信號sig被發送到進程pid.
如果pid等于0,那么信號sig被發送到所以和pid進程在同一個進程組的進程
如果pid等于-1,那么信號發給所有的進程表中的進程,除了最大的哪個進程號.
如果pid由于-1,和0一樣,只是發送進程組是-pid.
我們用最多的是第一個情況.還記得我們在守護進程那一節的例子嗎?我們那個時候用這個函數殺死了父進程守護進程的創建
raise系統調用向自己發送一個sig信號.我們可以用上面那個函數來實現這個功能的.
alarm函數和時間有點關系了,這個函數可以在seconds秒后向自己發送一個SIGALRM信號. 下面這個函數會有什么結果呢? 

#include  

main()
{
unsigned int i;
alarm(1);
for(i=0;1;i++)
printf("I=%d",i);
}
SIGALRM的缺省操作是結束進程,所以程序在1秒之后結束,你可以看看你的最后I值為多少,來比較一下大家的系統性能差異(我的是2232).

2。信號操作     有時候我們希望進程正確的執行,而不想進程受到信號的影響,比如我們希望上面那個程序在1秒鐘之后不結束.這個時候我們就要進行信號的操作了.
信號操作最常用的方法是信號屏蔽.信號屏蔽要用到下面的幾個函數. 

#include  

int sigemptyset(sigset_t *set);
int  sigfillset(sigset_t *set);
int sigaddset(sigset_t *set,int signo);
int sigdelset(sigset_t *set,int signo);
int sigismember(sigset_t *set,int signo);
int sigprocmask(int how,const sigset_t *set,sigset_t *oset);

sigemptyset函數初始化信號集合set,將set設置為空.sigfillset也初始化信號集合,只是將信號集合設置為所有信號的集合.sigaddset將信號signo加入到信號集合之中,sigdelset將信號從信號集合中刪除.sigismember查詢信號是否在信號集合之中.
sigprocmask是最為關鍵的一個函數.在使用之前要先設置好信號集合set.這個函數的作用是將指定的信號集合set加入到進程的信號阻塞集合之中去,如果提供了oset那么當前的進程信號阻塞集合將會保存在oset里面.參數how決定函數的操作方式. 
SIG_BLOCK:增加一個信號集合到當前進程的阻塞集合之中. 
SIG_UNBLOCK:從當前的阻塞集合之中刪除一個信號集合. 
SIG_SETMASK:將當前的信號集合設置為信號阻塞集合. 
以一個實例來解釋使用這幾個函數. 

#include  
#include  
#include  
#include  

int main(int argc,char **argv)
{
double y;
sigset_t intmask;
int i,repeat_factor;

if(argc!=2)
  {
fprintf(stderr,"Usage:%s repeat_factor\n\a",argv[0]);
exit(1);
  }

if((repeat_factor=atoi(argv[1]))<1)repeat_factor=10;
sigemptyset(&intmask);    /* 將信號集合設置為空  */ 
sigaddset(&intmask,SIGINT); /* 加入中斷 Ctrl+C 信號*/
while(1)
  {
/*阻塞信號,我們不希望保存原來的集合所以參數為NULL*/
sigprocmask(SIG_BLOCK,&intmask,NULL);
fprintf(stderr,"SIGINT signal blocked\n");
for(i=0;i fprintf(stderr,"Blocked calculation is finished\n");
/*  取消阻塞 */
sigprocmask(SIG_UNBLOCK,&intmask,NULL);
fprintf(stderr,"SIGINT signal unblocked\n");
for(i=0;i         fprintf(stderr,"Unblocked calculation is finished\n");
  }
exit(0);
}

程序在運行的時候我們要使用Ctrl+C來結束.如果我們在第一計算的時候發出SIGINT信號,由于信號已經屏蔽了,所以程序沒有反映.只有到信號被取消阻塞的時候程序才會結束. 注意我們只要發出一次SIGINT信號就可以了,因為信號屏蔽只是將信號加入到信號阻塞集合之中,并沒有丟棄這個信號.一旦信號屏蔽取消了,這個信號就會發生作用. 
有時候我們希望對信號作出及時的反映的,比如當擁護按下Ctrl+C時,我們不想什么事情也不做,我們想告訴用戶你的這個操作不好,請不要重試,而不是什么反映也沒有的. 這個時候我們要用到sigaction函數. 
#include  

   int sigaction(int signo,const struct sigaction *act,
struct sigaction *oact);

struct sigaction {
void (*sa_handler)(int signo);
void (*sa_sigaction)(int siginfo_t *info,void *act);
sigset_t sa_mask;
int  sa_flags;
void (*sa_restore)(void);
        } 

這個函數和結構看起來是不是有點恐怖呢.不要被這個嚇著了,其實這個函數的使用相當簡單的.我們先解釋一下各個參數的含義. signo很簡單就是我們要處理的信號了,可以是任何的合法的信號.有兩個信號不能夠使用(SIGKILL和SIGSTOP). act包含我們要對這個信號進行如何處理的信息.oact更簡單了就是以前對這個函數的處理信息了,主要用來保存信息的,一般用NULL就OK了. 
信號結構有點復雜.不要緊我們慢慢的學習. 
sa_handler是一個函數型指針,這個指針指向一個函數,這個函數有一個參數.這個函數就是我們要進行的信號操作的函數. sa_sigaction,sa_restore和sa_handler差不多的,只是參數不同罷了.這兩個元素我們很少使用,就不管了. 
sa_flags用來設置信號操作的各個情況.一般設置為0好了.sa_mask我們已經學習過了 
在使用的時候我們用sa_handler指向我們的一個信號操作函數,就可以了.sa_handler有兩個特殊的值:SIG_DEL和SIG_IGN.SIG_DEL是使用缺省的信號操作函數,而SIG_IGN是使用忽略該信號的操作函數. 
這個函數復雜,我們使用一個實例來說明.下面這個函數可以捕捉用戶的CTRL+C信號.并輸出一個提示語句. 

#include  
#include  
#include  
#include  
#include  

#define PROMPT "你想終止程序嗎?"

char *prompt=PROMPT;

void ctrl_c_op(int signo)
{
write(STDERR_FILENO,prompt,strlen(prompt));
}

int  main()
{
struct sigaction act;

act.sa_handler=ctrl_c_op;
sigemptyset(&act.sa_mask);
act.sa_flags=0;
if(sigaction(SIGINT,&act,NULL)<0)
  {
    fprintf(stderr,"Install Signal Action Error:%s\n\a",strerror(errno));
    exit(1);
  }
  while(1);
}

在上面程序的信號操作函數之中,我們使用了write函數而沒有使用fprintf函數.是因為我們要考慮到下面這種情況.如果我們在信號操作的時候又有一個信號發生,那么程序該如何運行呢? 為了處理在信號處理函數運行的時候信號的發生,我們需要設置sa_mask成員. 我們將我們要屏蔽的信號添加到sa_mask結構當中去,這樣這些函數在信號處理的時候就會被屏蔽掉的. 
3。其它信號函數     由于信號的操作和處理比較復雜,我們再介紹幾個信號操作函數. 

#include 
#include 

int pause(void);
int sigsuspend(const sigset_t *sigmask);

pause函數很簡單,就是掛起進程直到一個信號發生了.而sigsuspend也是掛起進程只是在調用的時候用sigmask取代當前的信號阻塞集合. 
#include 

int sigsetjmp(sigjmp_buf env,int val);
void  siglongjmp(sigjmp_buf env,int val);

還記得goto函數或者是setjmp和longjmp函數嗎.這兩個信號跳轉函數也可以實現程序的跳轉讓我們可以從函數之中跳轉到我們需要的地方. 
由于上面幾個函數,我們很少遇到,所以只是說明了一下,詳細情況請查看聯機幫助. 
4。一個實例     還記得我們在守護進程創建的哪個程序嗎?守護進程在這里我們把那個程序加強一下. 下面這個程序會在也可以檢查用戶的郵件.不過提供了一個開關,如果用戶不想程序提示有新的郵件到來,可以向程序發送SIGUSR2信號,如果想程序提供提示可以發送SIGUSR1信號. 


#include  
#include  
#include  
#include  
#include  
#include  
#include  

#include  
#include  

/*  Linux  的默任個人的郵箱地址是 /var/spool/mail/ */

#define  MAIL_DIR   "/var/spool/mail/"

/*      睡眠10秒鐘      */
                
#define  SLEEP_TIME     10
#define  MAX_FILENAME 255

unsigned char notifyflag=1;

long get_file_size(const char *filename)
{
  struct stat buf;

  if(stat(filename,&;buf)==-1)
   {
if(errno==ENOENT)return 0;
else return -1;
   }
  return (long)buf.st_size;
}

void send_mail_notify(void)
{
  fprintf(stderr,"New mail has arrived\007\n");
}

void turn_on_notify(int signo)
{
notifyflag=1;
}

void turn_off_notify(int signo)
{
notifyflag=0;
}

int check_mail(const char *filename)
{
  long old_mail_size,new_mail_size;
  sigset_t blockset,emptyset;

  sigemptyset(&;blockset);
  sigemptyset(&;emptyset);
  sigaddset(&;blockset,SIGUSR1);
  sigaddset(&;blockset,SIGUSR2);
  
  old_mail_size=get_file_size(filename);
  if(old_mail_size<0)return 1;
  if(old_mail_size>0) send_mail_notify();
  sleep(SLEEP_TIME);
  
  while(1)
  {
if(sigprocmask(SIG_BLOCK,&;blockset,NULL)<0) return 1;
while(notifyflag==0)sigsuspend(&;emptyset);
if(sigprocmask(SIG_SETMASK,&;emptyset,NULL)<0) return 1;
new_mail_size=get_file_size(filename);
if(new_mail_size>old_mail_size)send_mail_notify;
old_mail_size=new_mail_size;
sleep(SLEEP_TIME);
  }
}

int main(void)
{
  char mailfile[MAX_FILENAME];
  struct sigaction newact;
  struct passwd *pw;

  if((pw=getpwuid(getuid()))==NULL)
   {
fprintf(stderr,"Get Login Name Error:%s\n\a",strerror(errno));
exit(1);
   }
  strcpy(mailfile,MAIL_DIR);
  strcat(mailfile,pw->pw_name);
  newact.sa_handler=turn_on_notify;
  newact.sa_flags=0;
  sigemptyset(&;newact.sa_mask);
  sigaddset(&;newact.sa_mask,SIGUSR1);
  sigaddset(&;newact.sa_mask,SIGUSR2);
  if(sigaction(SIGUSR1,&;newact,NULL)<0)
   fprintf(stderr,"Turn On Error:%s\n\a",strerror(errno));
  newact.sa_handler=turn_off_notify;
  if(sigaction(SIGUSR1,&;newact,NULL)<0)
  fprintf(stderr,"Turn Off Error:%s\n\a",strerror(errno));
  check_mail(mailfile);
  exit(0);  
}

信號操作是一件非常復雜的事情,比我們想象之中的復雜程度還要復雜,如果你想徹底的弄清楚信號操作的各個問題,那么除了大量的練習以外還要多看聯機手冊.不過如果我們只是一般的使用的話,有了上面的幾個函數也就差不多了. 我們就介紹到這里了. 

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
国产成人啪免费观看软件 | 亚洲欧美激情视频在线观看一区二区三区 | 日韩精品中文字幕一区| 成人三级在线视频| 亚洲国产视频一区二区| 欧美激情综合网| 欧美一区二区三区色| 91亚洲男人天堂| 精品影视av免费| 午夜精品影院在线观看| 国产精品久久久久久久岛一牛影视 | 国产一区二区免费看| 亚洲永久精品大片| 国产精品久久久久久久久果冻传媒| 91精品国产综合久久精品图片| av动漫一区二区| 国产成人免费视频网站高清观看视频 | 色综合 综合色| 国产成人av一区二区三区在线| 日韩国产在线观看一区| 亚洲精品免费视频| 最新不卡av在线| 久久久99精品免费观看| 精品国产在天天线2019| 91精品国产欧美日韩| 欧美午夜片在线观看| 99精品视频免费在线观看| 国产伦精品一区二区三区免费| 日本91福利区| 日本伊人午夜精品| 亚洲第一狼人社区| 一区二区在线观看免费| 亚洲精品亚洲人成人网| 亚洲免费在线观看| 亚洲欧美日韩一区二区三区在线观看| 国产精品久久久久久久久动漫| 中文字幕精品一区| 中文字幕中文字幕一区| 国产精品欧美一区二区三区| 国产日产亚洲精品系列| 国产精品乱人伦中文| 国产区在线观看成人精品| 国产偷v国产偷v亚洲高清| 国产日产欧美一区| 中文字幕高清一区| 亚洲欧美色图小说| 亚洲一级在线观看| 日本在线不卡视频| 国产麻豆视频一区| 国产凹凸在线观看一区二区 | 欧美精品乱人伦久久久久久| 欧美美女一区二区三区| 欧美一区二区三区思思人| 日韩欧美一区二区视频| 久久综合久久综合九色| 久久精品亚洲精品国产欧美| 国产精品护士白丝一区av| 一区二区欧美国产| 日韩电影在线免费看| 免费精品视频在线| 国产麻豆精品一区二区| 波多野结衣亚洲一区| 色哟哟精品一区| 正在播放一区二区| 久久精品欧美日韩精品| 国产精品动漫网站| 亚洲第一主播视频| 国内精品免费**视频| aaa欧美色吧激情视频| 欧美性大战xxxxx久久久| 欧美一区二区三区播放老司机| 久久男人中文字幕资源站| 亚洲特级片在线| 日韩av在线发布| 不卡免费追剧大全电视剧网站| 欧美性大战久久久久久久蜜臀 | 91视频91自| 欧美精品成人一区二区三区四区| 精品国产区一区| 综合激情成人伊人| 欧美a级理论片| 99视频国产精品| 91精品国产入口在线| 中文一区二区完整视频在线观看| 亚洲午夜激情网站| 国产精品综合av一区二区国产馆| 日本国产一区二区| 久久这里只精品最新地址| 一区二区在线免费观看| 国产乱码精品一区二区三| 91久久香蕉国产日韩欧美9色| 欧美大片一区二区三区| 亚洲少妇30p| 国产一区二区三区黄视频| 欧美性一二三区| 欧美精彩视频一区二区三区| 日日摸夜夜添夜夜添国产精品| 岛国精品在线播放| 制服丝袜亚洲色图| 一区二区三区自拍| 懂色一区二区三区免费观看| 777精品伊人久久久久大香线蕉| 国产精品伦理一区二区| 韩国中文字幕2020精品| 欧美日韩在线精品一区二区三区激情 | 亚洲第一久久影院| av一区二区三区四区| 久久影院午夜片一区| 午夜精品在线看| 欧亚洲嫩模精品一区三区| 国产女主播在线一区二区| 蓝色福利精品导航| 欧美日韩一区二区电影| 亚洲欧美电影院| 成人免费毛片app| 久久精品亚洲精品国产欧美kt∨| 青青草成人在线观看| 欧美剧情片在线观看| 一区二区三区高清| 91视频在线观看| 综合中文字幕亚洲| 成人激情综合网站| 国产午夜亚洲精品理论片色戒| 久久99久久精品| 日韩精品一区二区三区在线观看 | 奇米影视一区二区三区| 欧美日韩国产欧美日美国产精品| 亚洲欧美成人一区二区三区| 99热99精品| 中文av一区特黄| 懂色av一区二区三区蜜臀| 久久精品视频一区二区| 国产精品一区久久久久| 精品久久久久久最新网址| 美女网站在线免费欧美精品| 欧美精品丝袜久久久中文字幕| 视频在线观看国产精品| 欧美日韩国产影片| 亚洲成a人片综合在线| 欧美少妇xxx| 视频一区视频二区中文字幕| 欧美日韩一区中文字幕| 婷婷成人激情在线网| 91麻豆精品国产自产在线观看一区 | 日韩一区二区视频| 美国三级日本三级久久99| 精品少妇一区二区三区日产乱码 | 欧美国产丝袜视频| 日韩欧美一区中文| 欧美日韩亚洲综合一区二区三区| 蜜臀av一级做a爰片久久| 亚洲品质自拍视频| 国产欧美精品在线观看| 久久久激情视频| 久久久久99精品国产片| 欧美大片在线观看| 日韩欧美一级精品久久| 久久先锋影音av鲁色资源网| 欧美日本韩国一区二区三区视频 | 久99久精品视频免费观看| 国产精品久久久久影院亚瑟| 欧美精品国产精品| 在线看不卡av| 欧美一区二区三区思思人| 亚洲啪啪综合av一区二区三区| 成人听书哪个软件好| 日本福利一区二区| 久久久综合视频| 日韩欧美国产一区二区三区 | 日韩欧美黄色影院| 欧美日韩国产免费| 717成人午夜免费福利电影| 成人激情综合网站| 粉嫩一区二区三区性色av| 精品一区二区在线播放| 免费在线观看一区二区三区| 亚洲一二三专区| 日韩专区中文字幕一区二区| 亚洲乱码中文字幕| 亚洲国产中文字幕在线视频综合| 亚洲国产高清aⅴ视频| 国产精品伦理在线| 亚洲在线观看免费视频| 亚洲va韩国va欧美va精品| 亚洲午夜羞羞片| 亚洲123区在线观看| 亚洲第四色夜色| 精品在线免费观看| 不卡的av网站| 欧美在线观看一区二区| 欧美成人猛片aaaaaaa| 国产视频911| 国产白丝精品91爽爽久久| 91福利在线免费观看| 国产精品国模大尺度视频| 亚洲欧美另类久久久精品| 国产v综合v亚洲欧| 久久久久久久久久久久久久久99| 久久福利视频一区二区| 精品欧美久久久|