国产成人精品三级麻豆,色综合天天综合高清网,亚洲精品夜夜夜,国产成人综合在线女婷五月99播放,色婷婷色综合激情国产日韩


信號處理函數(shù)signal()和信號集函數(shù)組

分享到:
           

    本文關(guān)鍵字: 信號處理函數(shù),signal(),信號集函數(shù)組

    信號處理的方法主要有兩種,一種是使用signal()函數(shù),另一種是使用信號集函數(shù)組。下面分別介紹這兩種處理方式。

    1)使用signal()函數(shù)

    使用signal()函數(shù)處理時,只需指出要處理的信號和處理函數(shù)即可。它主要用于前32種非實時信號的處理,不支持信號傳遞信息,但是由于使用簡單、易于理解,因此也受到很多程序員的歡迎。Linux還支持一個更健壯更新的信號處理函數(shù)sigaction(),推薦使用該函數(shù)。

    signal()函數(shù)的語法要點如表1所示。

表1 signal()函數(shù)語法要點

所需頭文件 #include <signal.h>
函數(shù)原型 typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
函數(shù)傳入值 signum:指定信號代碼
handler SIG_IGN:忽略該信號
SIG_DFL:采用系統(tǒng)默認(rèn)方式處理信號
自定義的信號處理函數(shù)指針
函數(shù)返回值 成功:以前的信號處理配置
出錯:-1

    這里需要對該函數(shù)原型進行說明。這個函數(shù)原型有點復(fù)雜:首先該函數(shù)原型整體指向一個無返回值并且?guī)б粋整型參數(shù)的函數(shù)指針,也就是信號的原始配置函數(shù);接著該原型又帶有兩個參數(shù),其中第2個參數(shù)可以是用戶自定義的信號處理函數(shù)的函數(shù)指針。

    表2列舉了sigaction()函數(shù)的語法要點。

表2 sigaction()函數(shù)語法要點

所需頭文件 #include <signal.h>
函數(shù)原型 int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact)
函數(shù)傳入值 signum:信號代碼,可以為除SIGKILL及SIGSTOP外的任何一個特定有效的信號
act:指向結(jié)構(gòu)sigaction的一個實例的指針,指定對特定信號的處理
oldact:保存原來對相應(yīng)信號的處理
函數(shù)返回值 成功:0
出錯:-1

    這里要說明的是sigaction()函數(shù)中第2和第3個參數(shù)用到的sigaction結(jié)構(gòu),這是一個看似非常復(fù)雜的結(jié)構(gòu),希望讀者能夠慢慢閱讀此段內(nèi)容。

    首先給出了sigaction的定義,代碼如下:

    struct sigaction
    {
        void (*sa_handler)(int signo);
        sigset_t sa_mask;
        int sa_flags;
        void (*sa_restore)(void);
    }

    sa_handler是一個函數(shù)指針,指定信號處理函數(shù),這里除可以是用戶自定義的處理函數(shù)外,還可以為SIG_DFL(采用默認(rèn)的處理方式)或SIG_IGN(忽略信號)。它的處理函數(shù)只有一個參數(shù),即信號值。

    sa_mask是一個信號集,它可以指定在信號處理程序執(zhí)行過程中哪些信號應(yīng)當(dāng)被屏蔽,在調(diào)用信號捕獲函數(shù)前,該信號集要加入到信號的信號屏蔽字中。

    sa_flags中包含了許多標(biāo)志位,是對信號進行處理的各個選擇項。它的常見可選值如表3所示。

表3 常見信號的含義及其默認(rèn)操作

信 號 含 義
SA_NODEFER / SA_NOMASK 當(dāng)捕捉到此信號時,在執(zhí)行其信號捕捉函數(shù)時,系統(tǒng)不會自動屏蔽此信號
SA_NOCLDSTOP 進程忽略子進程產(chǎn)生的任何SIGSTOP、SIGTSTP、SIGTTIN和SIGTTOU信號
SA_RESTART 令重啟的系統(tǒng)調(diào)用起作用
SA_ONESHOT / SA_RESETHAND 自定義信號只執(zhí)行一次,在執(zhí)行完畢后恢復(fù)信號的系統(tǒng)默認(rèn)動作

    以下實例表明了如何使用signal()函數(shù)捕捉相應(yīng)信號,并做出給定的處理。這里,my_func就是信號處理的函數(shù)指針,讀者還可以將其改為SIG_IGN或SIG_DFL查看運行結(jié)果。第2個實例是用sigaction()函數(shù)實現(xiàn)同樣的功能。

    以下是使用signal()函數(shù)的示例:

    /* signal.c */
    #include <signal.h>
    #include <stdio.h>
    #include <stdlib.h>

    /* 自定義信號處理函數(shù) */
    void my_func(int sign_no)
    {
        if (sign_no == SIGINT)
        {
            printf("I have get SIGINT\n");
        }
        else if (sign_no == SIGQUIT)
        {
            printf("I have get SIGQUIT\n");
        }
    }

    int main()
    {
        printf("Waiting for signal SIGINT or SIGQUIT...\n");

        /* 發(fā)出相應(yīng)的信號,并跳轉(zhuǎn)到信號處理函數(shù)處 */
        signal(SIGINT, my_func);
        signal(SIGQUIT, my_func);
        pause();
        exit(0);
    }

    運行結(jié)果如下:

    $ ./signal
    Waiting for signal SIGINT or SIGQUIT...
    I have get SIGINT (按Ctrl+c 組合鍵)
    $ ./signal
    Waiting for signal SIGINT or SIGQUIT...
    I have get SIGQUIT (按Ctrl+\ 組合鍵)

    以下是用sigaction()函數(shù)實現(xiàn)同樣的功能,只列出了更新的main()函數(shù)部分。

    /* sigaction.c */
    /* 前部分省略 */
    int main()
    {
        struct sigaction action;
        printf("Waiting for signal SIGINT or SIGQUIT...\n");

        /* sigaction結(jié)構(gòu)初始化 */
        action.sa_handler = my_func;
        sigemptyset(&action.sa_mask);
        action.sa_flags = 0;

        /* 發(fā)出相應(yīng)的信號,并跳轉(zhuǎn)到信號處理函數(shù)處 */
        sigaction(SIGINT, &action, 0);
        sigaction(SIGQUIT, &action, 0);
        pause();
        exit(0);
    }

    2)信號集函數(shù)組

    使用信號集函數(shù)組處理信號時涉及一系列的函數(shù),這些函數(shù)按照調(diào)用的先后次序可分為以下幾大功能模塊:創(chuàng)建信號集、注冊信號處理函數(shù)及檢測信號。

    其中,創(chuàng)建信號集主要用于處理用戶感興趣的一些信號,其函數(shù)包括以下幾個。

    ● sigemptyset():將信號集初始化為空。

    ● sigfillset():將信號集初始化為包含所有已定義的信號集。

    ● sigaddset():將指定信號加入到信號集中。

    ● sigdelset():將指定信號從信號集中刪除。

    ● sigismember():查詢指定信號是否在信號集中。

    注冊信號處理函數(shù)主要用于決定進程如何處理信號。這里要注意的是,信號集里的信號并不是真正可以處理的信號,只有當(dāng)信號的狀態(tài)處于非阻塞狀態(tài)時才會真正起作用。因此,首先使用sigprocmask()函數(shù)檢測并更改信號屏蔽字(信號屏蔽字是用來指定當(dāng)前被阻塞的一組信號,它們不會被進程接收),然后使用sigaction()函數(shù)來定義進程接收到特定信號后的行為。

    檢測信號是信號處理的后續(xù)步驟,因為被阻塞的信號不會傳遞給進程,所以這些信號就處于“未處理”狀態(tài)(也就是進程不清楚它的存在)。sigpending()函數(shù)允許進程檢測“未處理”信號,并進一步?jīng)Q定對它們做何處理。

    首先介紹創(chuàng)建信號集的函數(shù)格式,表4列舉了這一組函數(shù)的語法要點。

表4 創(chuàng)建信號集函數(shù)語法要點

所需頭文件 #include <signal.h>
函數(shù)原型 int sigemptyset(sigset_t *set)
int sigfillset(sigset_t *set)
int sigaddset(sigset_t *set, int signum)
int sigdelset(sigset_t *set, int signum)
int sigismember(sigset_t *set, int signum)
函數(shù)傳入值 set:信號集
signum:指定信號代碼
函數(shù)返回值 成功:0(sigismember成功返回1,失敗返回0)
出錯:-1

    表5列舉了sigprocmask()函數(shù)的語法要點。

表5 sigprocmask()函數(shù)語法要點

所需頭文件 #include <signal.h>
函數(shù)原型 int sigprocmask(int how, const sigset_t *set, sigset_t *oset)
函數(shù)傳入值 how:決定函數(shù)的操作方式 SIG_BLOCK:增加一個信號集到當(dāng)前進程的阻塞集中
SIG_UNBLOCK:從當(dāng)前的阻塞集中刪除一個信號集
SIG_SETMASK:將當(dāng)前的信號集設(shè)置為信號阻塞集
set:指定信號集
oset:信號屏蔽字
函數(shù)返回值 成功:0
出錯:-1

    此處,若set是一個非空指針,則參數(shù)how表示函數(shù)的操作方式;若how為空,則表示忽略此操作。

    表6列舉了sigpending()函數(shù)的語法要點。

表6 sigpending()函數(shù)語法要點

所需頭文件 #include <signal.h>
函數(shù)原型 int sigpending(sigset_t *set)
函數(shù)傳入值 set:要檢測的信號集
函數(shù)返回值 成功:0
出錯:-1

    總之,在處理信號時,一般遵循如圖1所示的操作流程。


圖1 一般的信號操作處理流程

    本文選自華清遠(yuǎn)見嵌入式培訓(xùn)教材《從實踐中學(xué)嵌入式Linux應(yīng)用程序開發(fā)》

   熱點鏈接:

   1、信號捕捉函數(shù)alarm()和pause()
   2、信號發(fā)送函數(shù)kill()和raise()
   3、Linux下的信號機制
   4、有名管道(FIFO)
   5、標(biāo)準(zhǔn)流管道

更多新聞>>