欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C/C++實(shí)現(xiàn)捕獲所有信號(hào)的示例詳解

 更新時(shí)間:2024年03月08日 16:15:26   作者:初級(jí)代碼游戲  
Linux的信號(hào)機(jī)制大部分情況下用不到,但是由于大部分信號(hào)的默認(rèn)處理是終止進(jìn)程,不正確處理會(huì)惹麻煩,所以我們來(lái)看看如何使用C/C++實(shí)現(xiàn)捕獲所有信號(hào)吧

一、原理

Linux的信號(hào)可能在你無(wú)法意識(shí)到的情況下發(fā)生。

比如socket網(wǎng)絡(luò)斷開(kāi),默認(rèn)情況下發(fā)送SIGPIPE給正在send/recv的進(jìn)程從而殺死進(jìn)程,忘記了處理這個(gè)就很麻煩,程序大部分時(shí)間很正常,偶爾奇怪地終止,可能要花很大力氣才會(huì)找到原因。

而Linux的信號(hào)是不帶任何數(shù)據(jù)的,知道一個(gè)連接斷開(kāi)又能怎么樣呢?又不知道是哪個(gè)連接斷開(kāi)(其實(shí)我們會(huì)根據(jù)send/recv的返回值處理的嘛)。

所以為了省力,我們可以在程序開(kāi)始處捕獲所有信號(hào),然后再根據(jù)實(shí)際運(yùn)行時(shí)的情況針對(duì)性處理(大部分情況根本不需要任何處理)。

關(guān)于信號(hào)的更多知識(shí)在后面的“信號(hào)描述”函數(shù)里。

二、基礎(chǔ)

處理信號(hào)最簡(jiǎn)單的方式就是調(diào)用signal函數(shù),這個(gè)函數(shù)設(shè)置信號(hào)處理方法并返回之前的信號(hào)處理方法(以便你可以在執(zhí)行自己的處理之前或之后調(diào)用原來(lái)的處理方法從而形成調(diào)用鏈)。

       #include <signal.h>
 
       typedef void (*sighandler_t)(int);
 
       sighandler_t signal(int signum, sighandler_t handler);

signal還有幾種形式,不過(guò)這種最簡(jiǎn)單了。 

三、代碼

捕獲全部信號(hào)的代碼

    int __all_sig_catch(int argc, char ** argv, int fun(int, char **))
    {
        signal(SIGABRT, sig_default);
        signal(SIGALRM, sig_default);
        signal(SIGBUS, sig_default);
        signal(SIGCHLD, sig_default);
        signal(SIGCONT, sig_default);
        signal(SIGFPE, sig_default);
        signal(SIGHUP, sig_default);
        signal(SIGILL, sig_default);
        //signal(SIGINT, sig_default);//ctrl-c
        signal(SIGIO, sig_default);
        signal(SIGIOT, sig_default);
        signal(SIGKILL, sig_default);
        signal(SIGPIPE, sig_default);
        signal(SIGPOLL, sig_default);
        signal(SIGPROF, sig_default);
        signal(SIGPWR, sig_default);
        signal(SIGQUIT, sig_default);
        signal(SIGSEGV, sig_default);
        signal(SIGSTOP, sig_default);
        signal(SIGSYS, sig_default);
        signal(SIGTERM, sig_default);
        signal(SIGTRAP, sig_default);
        signal(SIGTSTP, sig_default);
        signal(SIGTTIN, sig_default);
        signal(SIGTTOU, sig_default);
        signal(SIGURG, sig_default);
        signal(SIGUSR1, sig_default);
        signal(SIGUSR2, sig_default);
        signal(SIGVTALRM, sig_default);
        signal(SIGWINCH, sig_default);
        signal(SIGXCPU, sig_default);
        signal(SIGXFSZ, sig_default);
 
        int ret;
        try
        {
            ret = fun(argc, argv);
        }
        catch (...)
        {
            thelog << "未處理的異常發(fā)生" << ende;
            return __LINE__;
        }
        return ret;
    }

這個(gè)函數(shù)的調(diào)用方式是給main函數(shù)套一層殼,當(dāng)然你也可以把參數(shù)都去掉,只保留設(shè)置信號(hào)處理函數(shù)的那部分。

SIGINT沒(méi)有捕獲,這是ctrl-c產(chǎn)生的信號(hào),我確實(shí)需要這樣結(jié)束程序。

最后還捕獲了一下未處理的異常,這也是經(jīng)常疏忽的部分。如果需要生成core文件,可以調(diào)用abort()。

注意SIGKILL和SIGSTOP是無(wú)法捕獲的(雖然上面代碼里面有)。

調(diào)用代碼

int _main(int argc, char ** argv)
{
    ......
}
int main(int argc, char ** argv)
{
    return __all_sig_catch(argc, argv, _main);
}

信號(hào)處理函數(shù)

sig_default是設(shè)置的信號(hào)處理函數(shù),必須符合signal函數(shù)的要求:

    extern "C" void sig_default(int sig)
    {
        signal(sig, sig_default);
        cout << "pid=" << getpid() << " " << sigstr(sig) << endl;
    }

注意必須是extern "C",代碼里再次調(diào)用了signal設(shè)置信號(hào)處理函數(shù),這么做的原因我現(xiàn)在不是很確定,不過(guò)這么做起碼應(yīng)該不會(huì)有什么問(wèn)題。

嚴(yán)格說(shuō)應(yīng)該調(diào)用一下之前的處理函數(shù)的,不過(guò)這樣就會(huì)很復(fù)雜。因?yàn)槲覀冊(cè)趍ain函數(shù)開(kāi)始處設(shè)置,所有信號(hào)應(yīng)該是還沒(méi)有被設(shè)置過(guò)的。

信號(hào)描述

sigstr是輸出信號(hào)名稱的函數(shù):

    //信號(hào)的描述
    char const * sigstr(long sig)
    {
        switch(sig)
        {
        case SIGABRT    :    return "SIGABRT    進(jìn)程調(diào)用abort函數(shù),進(jìn)程非正常退出";
        case SIGALRM    :    return "SIGALRM    用alarm函數(shù)設(shè)置的timer超時(shí)或setitimer函數(shù)設(shè)置的interval timer超時(shí)";
        case SIGBUS     :    return "SIGBUS     某種特定的硬件異常,通常由內(nèi)存訪問(wèn)引起";
        case SIGCHLD    :    return "SIGCHLD    子進(jìn)程Terminate或Stop";
        case SIGCONT    :    return "SIGCONT    從stop中恢復(fù)運(yùn)行";
#ifndef _LINUXOS
        case SIGEMT     :    return "SIGEMT     和實(shí)現(xiàn)相關(guān)的硬件異常";
#endif
        case SIGFPE     :    return "SIGFPE     數(shù)學(xué)相關(guān)的異常,如被0除,浮點(diǎn)溢出,等等";
        case SIGHUP     :    return "SIGHUP     終端斷開(kāi)";
        case SIGILL     :    return "SIGILL     非法指令異常";
            //case SIGINFO    :    return "SIGINFO    BSD signal。由Status Key產(chǎn)生,通常是CTRL+T。發(fā)送給所有Foreground Group的進(jìn)程     ";
        case SIGINT     :    return "SIGINT     由Interrupt Key產(chǎn)生,通常是CTRL+C或者DELETE";
        case SIGIO      :    return "SIGIO      異步IO事件";
            //case SIGIOT     :    return "SIGIOT     實(shí)現(xiàn)相關(guān)的硬件異常,一般對(duì)應(yīng)SIGABRT                                              ";
        case SIGKILL    :    return "SIGKILL    強(qiáng)制中止";
        case SIGPIPE    :    return "SIGPIPE    在reader中止之后寫Pipe的時(shí)候發(fā)送";
            //case SIGPOLL    :    return "SIGPOLL    當(dāng)某個(gè)事件發(fā)送給Pollable Device的時(shí)候發(fā)送                                        ";
        case SIGPROF    :    return "SIGPROF    Setitimer指定的Profiling Interval Timer所產(chǎn)生";
        case SIGPWR     :    return "SIGPWR     和系統(tǒng)相關(guān)。和UPS相關(guān)。";
        case SIGQUIT    :    return "SIGQUIT    輸入Quit Key(CTRL+\\)";
        case SIGSEGV    :    return "SIGSEGV    非法內(nèi)存訪問(wèn)";
        case SIGSTOP    :    return "SIGSTOP    中止進(jìn)程";
        case SIGSYS     :    return "SIGSYS     非法系統(tǒng)調(diào)用";
        case SIGTERM    :    return "SIGTERM    請(qǐng)求中止進(jìn)程,kill命令缺省發(fā)送";
        case SIGTRAP    :    return "SIGTRAP    實(shí)現(xiàn)相關(guān)的硬件異常。一般是調(diào)試異常";
        case SIGTSTP    :    return "SIGTSTP    Suspend Key,一般是Ctrl+Z";
        case SIGTTIN    :    return "SIGTTIN    當(dāng)Background Group的進(jìn)程嘗試讀取Terminal的時(shí)候發(fā)送";
        case SIGTTOU    :    return "SIGTTOU    當(dāng)Background Group的進(jìn)程嘗試寫Terminal的時(shí)候發(fā)送";
        case SIGURG     :    return "SIGURG     當(dāng)out-of-band data接收的時(shí)候可能發(fā)送";
        case SIGUSR1    :    return "SIGUSR1    用戶自定義signal 1";
        case SIGUSR2    :    return "SIGUSR2    用戶自定義signal 2";
        case SIGVTALRM  :    return "SIGVTALRM  setitimer函數(shù)設(shè)置的Virtual Interval Timer超時(shí)的時(shí)候";
        case SIGWINCH   :    return "SIGWINCH   當(dāng)Terminal的窗口大小改變的時(shí)候,發(fā)送給Foreground Group的所有進(jìn)程";
        case SIGXCPU    :    return "SIGXCPU    當(dāng)CPU時(shí)間限制超時(shí)的時(shí)候";
        case SIGXFSZ    :    return "SIGXFSZ    進(jìn)程超過(guò)文件大小限制";
        default:             return "未知的信號(hào)";
        }
    }

其中屏蔽掉了一些是因?yàn)榧嫒菪詥?wèn)題(這段代碼以前要在IBM、SUN、HP的小型機(jī)上運(yùn)行,后來(lái)才改在Linux上運(yùn)行),你可以根據(jù)需要添加。

到此這篇關(guān)于C/C++實(shí)現(xiàn)捕獲所有信號(hào)的示例詳解的文章就介紹到這了,更多相關(guān)C++捕獲所有信號(hào)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++異常使用詳解(看這一篇就夠了)

    C++異常使用詳解(看這一篇就夠了)

    C++中的異常是指在程序執(zhí)行過(guò)程中發(fā)生錯(cuò)誤,導(dǎo)致程序無(wú)法正常運(yùn)行的情況,下面這篇文章主要給大家介紹了關(guān)于C++異常使用的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-10-10
  • [c++]變量聲明與定義的規(guī)則詳解

    [c++]變量聲明與定義的規(guī)則詳解

    這篇文章主要介紹了[c++]變量聲明與定義的規(guī)則詳解,對(duì)于學(xué)習(xí)c++的朋友來(lái)說(shuō)這是一個(gè)很細(xì)膩的文章,代碼完整,需要的朋友可以參考下
    2021-04-04
  • C++使用boost::lexical_cast進(jìn)行數(shù)值轉(zhuǎn)換

    C++使用boost::lexical_cast進(jìn)行數(shù)值轉(zhuǎn)換

    這篇文章介紹了C++使用boost::lexical_cast進(jìn)行數(shù)值轉(zhuǎn)換的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-06-06
  • C++之Primer算術(shù)運(yùn)算符詳解

    C++之Primer算術(shù)運(yùn)算符詳解

    這篇文章主要介紹了C++之Primer算術(shù)運(yùn)算符方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2025-03-03
  • C語(yǔ)言大小端模式、判斷大小端、大小端轉(zhuǎn)換方法詳解

    C語(yǔ)言大小端模式、判斷大小端、大小端轉(zhuǎn)換方法詳解

    這篇文章主要介紹了C語(yǔ)言大小端模式、判斷大小端、大小端轉(zhuǎn)換的相關(guān)資料,大端和小端是數(shù)據(jù)在內(nèi)存中的存儲(chǔ)方式,大端模式下高字節(jié)存于低地址,小端模式則相反,大小端問(wèn)題由數(shù)據(jù)類型多字節(jié)存儲(chǔ)引起,不同選擇形成不同存儲(chǔ)模式,需要的朋友可以參考下
    2024-10-10
  • c++代碼實(shí)現(xiàn)tea加密算法的實(shí)例詳解

    c++代碼實(shí)現(xiàn)tea加密算法的實(shí)例詳解

    這篇文章主要介紹了c++代碼實(shí)現(xiàn)tea加密算法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-04-04
  • C語(yǔ)言數(shù)組的各種操作梳理

    C語(yǔ)言數(shù)組的各種操作梳理

    數(shù)組是一組有序的數(shù)據(jù)的集合,數(shù)組中元素類型相同,由數(shù)組名和下標(biāo)唯一地確定,數(shù)組中數(shù)據(jù)不僅數(shù)據(jù)類型相同,而且在計(jì)算機(jī)內(nèi)存里連續(xù)存放,地址編號(hào)最低的存儲(chǔ)單元存放數(shù)組的起始元素,地址編號(hào)最高的存儲(chǔ)單元存放數(shù)組的最后一個(gè)元素
    2022-04-04
  • C語(yǔ)言實(shí)現(xiàn)數(shù)組元素排序方法詳解

    C語(yǔ)言實(shí)現(xiàn)數(shù)組元素排序方法詳解

    這篇文章主要為大家介紹了C語(yǔ)言算法練習(xí)中數(shù)組元素排序的實(shí)現(xiàn)方法,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C語(yǔ)言有一定幫助,需要的可以參考一下
    2023-02-02
  • Qt圖形圖像開(kāi)發(fā)之Qt曲線圖美化QChart QScatterSeries 空心點(diǎn)陣圖,鼠標(biāo)移動(dòng)到上面顯示數(shù)值,鼠標(biāo)移開(kāi)數(shù)值消失效果實(shí)例

    Qt圖形圖像開(kāi)發(fā)之Qt曲線圖美化QChart QScatterSeries 空心點(diǎn)陣圖,鼠標(biāo)移動(dòng)到上面顯示數(shù)值,鼠標(biāo)移開(kāi)

    這篇文章主要介紹了Qt圖形圖像開(kāi)發(fā)之Qt曲線圖美化QChart QScatterSeries 空心點(diǎn)陣圖,鼠標(biāo)移動(dòng)到上面顯示數(shù)值,鼠標(biāo)移開(kāi)數(shù)值消失效果實(shí)例,需要的朋友可以參考下
    2020-03-03
  • OpenCV圖像處理之實(shí)現(xiàn)圖像膨脹腐蝕操作

    OpenCV圖像處理之實(shí)現(xiàn)圖像膨脹腐蝕操作

    圖像形態(tài)學(xué)操作是指基于形狀的一系列圖像處理操作的合集,主要是基于集合論基礎(chǔ)上的形態(tài)學(xué)數(shù)學(xué)對(duì)圖像進(jìn)行處理。本文將為大家介紹一下如何利用OpenCV實(shí)現(xiàn)其中的腐蝕和膨脹操作,需要的可以參考一下
    2022-09-09

最新評(píng)論