• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Linux程序設(shè)計(jì)入門--信號處理

            Linux程序設(shè)計(jì)入門--信號處理
            Linux下的信號事件
            前言:這一章我們討論一下Linux下的信號處理函數(shù).
            Linux下的信號處理函數(shù):
            信號的產(chǎn)生
            信號的處理
            其它信號函數(shù)
            一個實(shí)例

            1。信號的產(chǎn)生
            Linux下的信號可以類比于DOS下的INT或者是Windows下的事件.在有一個信號發(fā)生時
            候相信的信號就會發(fā)送給相應(yīng)的進(jìn)程.在Linux下的信號有以下幾個. 我們使用 kill -l
            命令可以得到以下的輸出結(jié)果:
            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
            關(guān)于這些信號的詳細(xì)解釋請查看man 7 signal的輸出結(jié)果. 信號事件的發(fā)生有兩個來源
            :一個是硬件的原因(比如我們按下了鍵盤),一個是軟件的原因(比如我們使用系統(tǒng)函數(shù)或
            者是命令發(fā)出信號). 最常用的四個發(fā)出信號的系統(tǒng)函數(shù)是kill, raise, alarm和setit
            imer函數(shù). setitimer函數(shù)我們在計(jì)時器的使用 那一章再學(xué)習(xí).
            1. #include <sys/types.h>    
            2. #include <signal.h>    
            3. #include <unistd.h>    
            4. int kill(pid_t pid,int sig);    
            5. int raise(int sig);    
            6. unisigned int alarm(unsigned int seconds);    

            kill系統(tǒng)調(diào)用負(fù)責(zé)向進(jìn)程發(fā)送信號sig.
            如果pid是正數(shù),那么向信號sig被發(fā)送到進(jìn)程pid.
            如果pid等于0,那么信號sig被發(fā)送到所以和pid進(jìn)程在同一個進(jìn)程組的進(jìn)程
            如果pid等于-1,那么信號發(fā)給所有的進(jìn)程表中的進(jìn)程,除了最大的哪個進(jìn)程號.
            如果pid由于-1,和0一樣,只是發(fā)送進(jìn)程組是-pid.
            我們用最多的是第一個情況.還記得我們在守護(hù)進(jìn)程那一節(jié)的例子嗎?我們那個時候用這
            個函數(shù)殺死了父進(jìn)程守護(hù)進(jìn)程的創(chuàng)建
            raise系統(tǒng)調(diào)用向自己發(fā)送一個sig信號.我們可以用上面那個函數(shù)來實(shí)現(xiàn)這個功能的.
            alarm函數(shù)和時間有點(diǎn)關(guān)系了,這個函數(shù)可以在seconds秒后向自己發(fā)送一個SIGALRM信號
            .. 下面這個函數(shù)會有什么結(jié)果呢?
             
            1. #include <unistd.h>    
            2. main()    
            3. {    
            4. unsigned int i;    
            5. alarm(1);    
            6. for(i=0;1;i++)    
            7. printf("I=%d",i);    
            8. }    

            SIGALRM的缺省操作是結(jié)束進(jìn)程,所以程序在1秒之后結(jié)束,你可以看看你的最后I值為多少
            ,來比較一下大家的系統(tǒng)性能差異(我的是2232).
            2。信號操作 有時候我們希望進(jìn)程正確的執(zhí)行,而不想進(jìn)程受到信號的影響,比如我
            們希望上面那個程序在1秒鐘之后不結(jié)束.這個時候我們就要進(jìn)行信號的操作了.
            信號操作最常用的方法是信號屏蔽.信號屏蔽要用到下面的幾個函數(shù).
             
            1. #include <signal.h>    
            2. int sigemptyset(sigset_t *set);    
            3. int sigfillset(sigset_t *set);    
            4. int sigaddset(sigset_t *set,int signo);    
            5. int sigdelset(sigset_t *set,int signo);    
            6. int sigismember(sigset_t *set,int signo);    
            7. int sigprocmask(int how,const sigset_t *set,sigset_t *oset);    

            sigemptyset函數(shù)初始化信號集合set,將set設(shè)置為空.sigfillset也初始化信號集合,只
            是將信號集合設(shè)置為所有信號的集合.sigaddset將信號signo加入到信號集合之中,sigd
            elset將信號從信號集合中刪除.sigismember查詢信號是否在信號集合之中.
            sigprocmask是最為關(guān)鍵的一個函數(shù).在使用之前要先設(shè)置好信號集合set.這個函數(shù)的作
            用是將指定的信號集合set加入到進(jìn)程的信號阻塞集合之中去,如果提供了oset那么當(dāng)前
            的進(jìn)程信號阻塞集合將會保存在oset里面.參數(shù)how決定函數(shù)的操作方式.
            SIG_BLOCK:增加一個信號集合到當(dāng)前進(jìn)程的阻塞集合之中.
            SIG_UNBLOCK:從當(dāng)前的阻塞集合之中刪除一個信號集合.
            SIG_SETMASK:將當(dāng)前的信號集合設(shè)置為信號阻塞集合.
            以一個實(shí)例來解釋使用這幾個函數(shù).
             
            1. #include <signal.h>    
            2. #include <stdio.h>    
            3. #include <math.h>    
            4. #include <stdlib.h>    
            5. int main(int argc,char **argv)    
            6. {    
            7. double y;    
            8. sigset_t intmask;    
            9. int i,repeat_factor;    
            10. if(argc!=2)    
            11. {    
            12. fprintf(stderr,"Usage:%s repeat_factor\n\a",argv[0]);    
            13. exit(1);    
            14. }    
            15. if((repeat_factor=atoi(argv[1]))<1)repeat_factor=10;    
            16. sigemptyset(&intmask); /* 將信號集合設(shè)置為空 */    
            17. sigaddset(&intmask,SIGINT); /* 加入中斷 Ctrl+C 信號*/    
            18. while(1)    
            19. {    
            20. /*阻塞信號,我們不希望保存原來的集合所以參數(shù)為NULL*/    
            21. sigprocmask(SIG_BLOCK,&intmask,NULL);    
            22. fprintf(stderr,"SIGINT signal blocked\n");    
            23. for(i=0;i<repeat_factor;i++)y=sin((double)i);    
            24. fprintf(stderr,"Blocked calculation is finished\n");    
            25. /* 取消阻塞 */    
            26. sigprocmask(SIG_UNBLOCK,&intmask,NULL);    
            27. fprintf(stderr,"SIGINT signal unblocked\n");    
            28. for(i=0;i<repeat_factor;i++)y=sin((double)i);    
            29. fprintf(stderr,"Unblocked calculation is finished\n");    
            30. }    
            31. exit(0);    
            32. }    

            程序在運(yùn)行的時候我們要使用Ctrl+C來結(jié)束.如果我們在第一計(jì)算的時候發(fā)出SIGINT信號
            ,由于信號已經(jīng)屏蔽了,所以程序沒有反映.只有到信號被取消阻塞的時候程序才會結(jié)束.
            注意我們只要發(fā)出一次SIGINT信號就可以了,因?yàn)樾盘柶帘沃皇菍⑿盘柤尤氲叫盘栕枞?
            集合之中,并沒有丟棄這個信號.一旦信號屏蔽取消了,這個信號就會發(fā)生作用.
            有時候我們希望對信號作出及時的反映的,比如當(dāng)擁護(hù)按下Ctrl+C時,我們不想什么事情
            也不做,我們想告訴用戶你的這個操作不好,請不要重試,而不是什么反映也沒有的. 這個
            時候我們要用到sigaction函數(shù).
             
            1. #include <signal.h>    
            2.   
            3. int sigaction(int signo,const struct sigaction *act,    
            4. struct sigaction *oact);    
            5. struct sigaction {    
            6. void (*sa_handler)(int signo);    
            7. void (*sa_sigaction)(int siginfo_t *info,void *act);    
            8. sigset_t sa_mask;    
            9. int sa_flags;    
            10. void (*sa_restore)(void);    
            11. }    

            這個函數(shù)和結(jié)構(gòu)看起來是不是有點(diǎn)恐怖呢.不要被這個嚇著了,其實(shí)這個函數(shù)的使用相當(dāng)
            簡單的.我們先解釋一下各個參數(shù)的含義. signo很簡單就是我們要處理的信號了,可以是
            任何的合法的信號.有兩個信號不能夠使用(SIGKILL和SIGSTOP). act包含我們要對這個
            信號進(jìn)行如何處理的信息.oact更簡單了就是以前對這個函數(shù)的處理信息了,主要用來保
            存信息的,一般用NULL就OK了.
            信號結(jié)構(gòu)有點(diǎn)復(fù)雜.不要緊我們慢慢的學(xué)習(xí).
            sa_handler是一個函數(shù)型指針,這個指針指向一個函數(shù),這個函數(shù)有一個參數(shù).這個函數(shù)就
            是我們要進(jìn)行的信號操作的函數(shù). sa_sigaction,sa_restore和sa_handler差不多的,只
            是參數(shù)不同罷了.這兩個元素我們很少使用,就不管了.
            sa_flags用來設(shè)置信號操作的各個情況.一般設(shè)置為0好了.sa_mask我們已經(jīng)學(xué)習(xí)過了
            在使用的時候我們用sa_handler指向我們的一個信號操作函數(shù),就可以了.sa_handler有
            兩個特殊的值:SIG_DEL和SIG_IGN.SIG_DEL是使用缺省的信號操作函數(shù),而SIG_IGN是使用
            忽略該信號的操作函數(shù).
            這個函數(shù)復(fù)雜,我們使用一個實(shí)例來說明.下面這個函數(shù)可以捕捉用戶的CTRL+C信號.并輸
            出一個提示語句.
             
            1. #include <signal.h>    
            2. #include <stdio.h>    
            3. #include <string.h>    
            4. #include <errno.h>    
            5. #include <unistd.h>    
            6. #define PROMPT "你想終止程序嗎?"    
            7. char *prompt=PROMPT;    
            8. void ctrl_c_op(int signo)    
            9. {    
            10. write(STDERR_FILENO,prompt,strlen(prompt));    
            11. }    
            12. int main()    
            13. {    
            14. struct sigaction act;    
            15. act.sa_handler=ctrl_c_op;    
            16. sigemptyset(&act.sa_mask);    
            17. act.sa_flags=0;    
            18. if(sigaction(SIGINT,&act,NULL)<0)    
            19. {    
            20. fprintf(stderr,"Install Signal Action Error:%s\n\a",strerror(errno));    
            21. exit(1);    
            22. }    
            23. while(1);    
            24. }    

            在上面程序的信號操作函數(shù)之中,我們使用了write函數(shù)而沒有使用fprintf函數(shù).是因?yàn)?
            我們要考慮到下面這種情況.如果我們在信號操作的時候又有一個信號發(fā)生,那么程序該
            如何運(yùn)行呢? 為了處理在信號處理函數(shù)運(yùn)行的時候信號的發(fā)生,我們需要設(shè)置sa_mask成
            員. 我們將我們要屏蔽的信號添加到sa_mask結(jié)構(gòu)當(dāng)中去,這樣這些函數(shù)在信號處理的時
            候就會被屏蔽掉的.
            3。其它信號函數(shù) 由于信號的操作和處理比較復(fù)雜,我們再介紹幾個信號操作函數(shù).
             
            1. #include <unistd.h>    
            2. #include <signal.h>    
            3. int pause(void);    
            4. int sigsuspend(const sigset_t *sigmask);    

            pause函數(shù)很簡單,就是掛起進(jìn)程直到一個信號發(fā)生了.而sigsuspend也是掛起進(jìn)程只是在
            調(diào)用的時候用sigmask取代當(dāng)前的信號阻塞集合.
            1. #include <sigsetjmp>    
            2. int sigsetjmp(sigjmp_buf env,int val);    
            3. void siglongjmp(sigjmp_buf env,int val);    

            還記得goto函數(shù)或者是setjmp和longjmp函數(shù)嗎.這兩個信號跳轉(zhuǎn)函數(shù)也可以實(shí)現(xiàn)程序的
            跳轉(zhuǎn)讓我們可以從函數(shù)之中跳轉(zhuǎn)到我們需要的地方.
            由于上面幾個函數(shù),我們很少遇到,所以只是說明了一下,詳細(xì)情況請查看聯(lián)機(jī)幫助.
            4。一個實(shí)例 還記得我們在守護(hù)進(jìn)程創(chuàng)建的哪個程序嗎?守護(hù)進(jìn)程在這里我們把那個
            程序加強(qiáng)一下. 下面這個程序會在也可以檢查用戶的郵件.不過提供了一個開關(guān),如果用
            戶不想程序提示有新的郵件到來,可以向程序發(fā)送SIGUSR2信號,如果想程序提供提示可以
            發(fā)送SIGUSR1信號.
             
            1. #include <unistd.h>    
            2. #include <stdio.h>    
            3. #include <errno.h>    
            4. #include <fcntl.h>    
            5. #include <signal.h>    
            6. #include <string.h>    
            7. #include <pwd.h>    
            8. #include <sys/types.h>    
            9. #include <sys/stat.h>    
            10. /* Linux 的默任個人的郵箱地址是 /var/spool/mail/ */    
            11. #define MAIL_DIR "/var/spool/mail/"    
            12. /* 睡眠10秒鐘 */    
            13. #define SLEEP_TIME 10    
            14. #define MAX_FILENAME 255    
            15. unsigned char notifyflag=1;    
            16. long get_file_size(const char *filename)    
            17. {    
            18. struct stat buf;    
            19. if(stat(filename,&;buf)==-1)    
            20. {    
            21. if(errno==ENOENT)return 0;    
            22. else return -1;    
            23. }    
            24. return (long)buf.st_size;    
            25. }    
            26. void send_mail_notify(void)    
            27. {    
            28. fprintf(stderr,"New mail has arrived\007\n");    
            29. }    
            30. void turn_on_notify(int signo)    
            31. {    
            32. notifyflag=1;    
            33. }    
            34. void turn_off_notify(int signo)    
            35. {    
            36. notifyflag=0;    
            37. }    
            38. int check_mail(const char *filename)    
            39. {    
            40. long old_mail_size,new_mail_size;    
            41. sigset_t blockset,emptyset;    
            42. sigemptyset(&;blockset);    
            43. sigemptyset(&;emptyset);    
            44. sigaddset(&;blockset,SIGUSR1);    
            45. sigaddset(&;blockset,SIGUSR2);    
            46. old_mail_size=get_file_size(filename);    
            47. if(old_mail_size<0)return 1;    
            48. if(old_mail_size>0) send_mail_notify();    
            49. sleep(SLEEP_TIME);    
            50. while(1)    
            51. {    
            52. if(sigprocmask(SIG_BLOCK,&;blockset,NULL)<0) return 1;    
            53. while(notifyflag==0)sigsuspend(&;emptyset);    
            54. if(sigprocmask(SIG_SETMASK,&;emptyset,NULL)<0) return 1;    
            55. new_mail_size=get_file_size(filename);    
            56. if(new_mail_size>old_mail_size)send_mail_notify;    
            57. old_mail_size=new_mail_size;    
            58. sleep(SLEEP_TIME);    
            59. }    
            60. }    
            61. int main(void)    
            62. {    
            63. char mailfile[MAX_FILENAME];    
            64. struct sigaction newact;    
            65. struct passwd *pw;    
            66. if((pw=getpwuid(getuid()))==NULL)    
            67. {    
            68. fprintf(stderr,"Get Login Name Error:%s\n\a",strerror(errno));    
            69. exit(1);    
            70. }    
            71. strcpy(mailfile,MAIL_DIR);    
            72. strcat(mailfile,pw->pw_name);    
            73. newact.sa_handler=turn_on_notify;    
            74. newact.sa_flags=0;    
            75. sigemptyset(&;newact.sa_mask);    
            76. sigaddset(&;newact.sa_mask,SIGUSR1);    
            77. sigaddset(&;newact.sa_mask,SIGUSR2);    
            78. if(sigaction(SIGUSR1,&;newact,NULL)<0)    
            79. fprintf(stderr,"Turn On Error:%s\n\a",strerror(errno));    
            80. newact.sa_handler=turn_off_notify;    
            81. if(sigaction(SIGUSR1,&;newact,NULL)<0)    
            82. fprintf(stderr,"Turn Off Error:%s\n\a",strerror(errno));    
            83. check_mail(mailfile);    
            84. exit(0);    
            85. }    

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

            posted on 2008-04-16 09:12 RedLight 閱讀(385) 評論(0)  編輯 收藏 引用 所屬分類: Linux(C++開發(fā)) (rss)

            <2008年9月>
            31123456
            78910111213
            14151617181920
            21222324252627
            2829301234
            567891011

            導(dǎo)航

            統(tǒng)計(jì)

            公告


            Name: Galen
            QQ: 88104725

            常用鏈接

            留言簿(3)

            隨筆分類

            隨筆檔案

            相冊

            My Friend

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            久久久久免费精品国产| 久久免费高清视频| 国内精品欧美久久精品| 国产成人精品久久| 久久婷婷五月综合国产尤物app | 日日躁夜夜躁狠狠久久AV| 久久精品国产男包| 91精品无码久久久久久五月天| 热综合一本伊人久久精品| 人妻精品久久久久中文字幕69| 国产精品美女久久久| 亚洲精品国精品久久99热| 国产成人无码精品久久久免费| 亚洲日本久久久午夜精品| 久久久久久国产精品免费免费| 久久99国产精品久久99| 天天躁日日躁狠狠久久| 久久婷婷五月综合成人D啪| 国产精品伊人久久伊人电影| 亚洲色大成网站www久久九| 久久AV无码精品人妻糸列| 久久福利青草精品资源站| 一本色道久久HEZYO无码| 99久久亚洲综合精品网站| 精品综合久久久久久97| 国产三级精品久久| 久久96国产精品久久久| 精品国产福利久久久| 综合网日日天干夜夜久久 | 国产午夜福利精品久久2021| 亚洲∧v久久久无码精品| 久久九九免费高清视频| 伊人丁香狠狠色综合久久| 精品人妻久久久久久888| 亚洲狠狠婷婷综合久久久久| 色综合合久久天天给综看| 国产免费久久精品丫丫| 久久艹国产| 久久久久亚洲av成人无码电影| 久久www免费人成精品香蕉| 久久精品国产欧美日韩|