• <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>

            woaidongmao

            文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
            隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
            數據加載中……

            C++程序的異常處理技巧

              處理 C++ 中的異常會在語言級別上遇到少許隱含限制,但在某些情況下,您可以繞過它們。學習各種利用異常的方法,您就可以生產更可靠的應用程序。

              保留異常來源信息

              在C++中,無論何時在處理程序內捕獲一個異常,關于該異常來源的信息都是不為人知的。異常的具體來源可以提供許多更好地處理該異常的重要信息,或者提供一些可以附加到錯誤日志的信息,以便以后進行分析。

              為了解決這一問題,可以在拋出異常語句期間,在異常對象的構造函數中生成一個堆棧跟蹤。ExceptionTracer是示范這種行為的一個類。

              清單 1. 在異常對象構造函數中生成一個堆棧跟蹤

              // Sample Program:

              // Compiler: gcc 3.2.3 20030502

              // Linux: Red Hat

              #include

              #include

              #include

              #include

              using namespace std;

              /////////////////////////////////////////////

              class ExceptionTracer

              {

              public:

              ExceptionTracer()

              {

              void * array[25];

              int nSize = backtrace(array, 25);

              char ** symbols = backtrace_symbols(array, nSize);

               

              for (int i = 0; i < nSize; i++)

              {

              cout << symbols[i] << endl;

              }

              free(symbols);

              }

              };

              管理信號

              每當進程執行一個令人討厭的動作,以致于 Linux? 內核發出一個信號時,該信號都必須被處理。信號處理程序通常會釋放一些重要資源并終止應用程序。在這種情況下,堆棧上的所有對象實例都處于未破壞狀態。另一方面,如果這些信號被轉換成C++ 異常,那么您可以優雅地調用其構造函數,并安排多層 catch 塊,以便更好地處理這些信號。

              清單 2 中定義的 SignalExceptionClass,提供了表示內核可能發出信號的 C++ 異常的抽象。SignalTranslator 是一個基于 SignalExceptionClass 的模板類,它通常用來實現到 C++ 異常的轉換。在任何瞬間,只能有一個信號處理程序處理一個活動進程的一個信號。因此,SignalTranslator 采用了 singleton 設計模式。整體概念通過用于 SIGSEGV SegmentationFault 類和用于 SIGFPE FloatingPointException 類得到了展示。

              清單 2. 將信號轉換成異常

              template class SignalTranslator

              {

              private:

              class SingleTonTranslator

              {

              public:

              SingleTonTranslator()

              {

              signal(SignalExceptionClass::GetSignalNumber(),

              SignalHandler);

              }

              static void SignalHandler(int)

              {

              throw SignalExceptionClass();

              }

              };

              public:

              SignalTranslator()

              {

              static SingleTonTranslator s_objTranslator;

              }

              };

              // An example for SIGSEGV

              class SegmentationFault : public ExceptionTracer, public

              exception

              {

              public:

              static int GetSignalNumber() {return SIGSEGV;}

              };

              SignalTranslator

              g_objSegmentationFaultTranslator;

              // An example for SIGFPE

              class FloatingPointException : public ExceptionTracer, public

              exception

              {

              public:

              static int GetSignalNumber() {return SIGFPE;}

              };

              SignalTranslator

              g_objFloatingPointExceptionTranslator;

              管理構造函數和析構函數中的異常

              在全局(靜態全局)變量的構造和析構期間,每個 ANSI C++ 都捕獲到異常是不可能的。因此,ANSI C++ 不建議在那些其實例可能被定義為全局實例(靜態全局實例)的類的構造函數和析構函數中拋出異常。換一種說法就是永遠都不要為那些其構造函數和析構函數可能拋出異常的類定義全局(靜態全局)實例。不過,如果假定有一個特定編譯器和一個特定系統,那么可能可以這樣做,幸運的是,對于Linux 上的 GCC,恰好是這種情況。

              使用 ExceptionHandler 類可以展示這一點,該類也采用了 singleton 設計模式。其構造函數注冊了一個未捕獲的處理程序。因為每次只能有一個未捕獲的處理程序處理一個活動進程,構造函數應該只被調用一次,因此要采用 singleton 模式。應該在定義有問題的實際全局(靜態全局)變量之前定義 ExceptionHandler 的全局(靜態全局)實例。

              清單 3. 處理構造函數中的異常

              class ExceptionHandler

              {

              private:

              class SingleTonHandler

              {

              public:

              SingleTonHandler()

              {

              set_terminate(Handler);

              }

              static void Handler()

              {

              // Exception from construction/destruction of global variables try

              {

              // re-throw throw;

              }

              catch (SegmentationFault &)

              {

              cout << SegmentationFault << endl;

              }

              catch (FloatingPointException &)

              {

              cout << FloatingPointException << endl;

              }

              catch (...)

              {

              cout << Unknown Exception << endl;

              }

              //if this is a thread performing some core activity

              abort();

              // else if this is a thread used to service requests

              // pthread_exit();

              }

              };

              public:

              ExceptionHandler()

              {

              static SingleTonHandler s_objHandler;

              }

              };

              //////////////////////////////////////////////////////////////////////////

              class A

              {

              public:

              A()

              {

              //int i = 0, j = 1/i;

              *(int *)0 = 0;

              }

              };

              // Before defining any global variable, we define a dummy instance

              // of ExceptionHandler object to make sure that

              // ExceptionHandler::SingleTonHandler::SingleTonHandler() is

              invoked

              ExceptionHandler g_objExceptionHandler;

              A g_a;

              //////////////////////////////////////////////////////////////////////////

              int main(int argc, char* argv[])

              {

              return 0;

              }

              處理多線程程序中的異常

              有時一些異常沒有被捕獲,這將造成進程異常中止。不過很多時候,進程包含多個線程,其中少數線程執行核心應用程序邏輯,同時,其余線程為外部請求提供服務。如果服務線程因編程錯誤而沒有處理某個異常,則會造成整個應用程序崩潰。這一點可能是不受人們歡迎的,因為它會通過向應用程序傳送不合法的請求而助長拒絕服務攻擊。為了避免這一點,未捕獲處理程序可以決定是請求異常中止調用,還是請求線程退出調用。清單3 ExceptionHandler::SingleTonHandler::Handler() 函數的末尾處展示了該處理程序。

              結束語

              我簡單地討論了少許 C++ 編程設計模式,以便更好地執行以下任務:

              ·在拋出異常的時候追蹤異常的來源。

              ·將信號從內核程序轉換成 C++ 異常。

              ·捕獲構造和/或析構全局變量期間拋出的異常。

              ·多線程進程中的異常處理。

             

            posted on 2008-11-09 18:08 肥仔 閱讀(375) 評論(0)  編輯 收藏 引用 所屬分類: C++ 基礎

            久久精品无码专区免费青青| av国内精品久久久久影院| 久久国产成人午夜aⅴ影院| 国产精品伦理久久久久久| 久久无码精品一区二区三区| 漂亮人妻被中出中文字幕久久 | 麻豆成人久久精品二区三区免费 | 久久人人爽人人爽人人片AV不 | 2021最新久久久视精品爱| 久久婷婷五月综合色奶水99啪| 热re99久久精品国产99热| 欧美性猛交xxxx免费看久久久| 午夜久久久久久禁播电影| 久久996热精品xxxx| 久久人妻少妇嫩草AV无码专区| 久久久久久久综合日本亚洲| 热久久国产欧美一区二区精品| 久久久久久无码Av成人影院| 欧美日韩中文字幕久久久不卡| 婷婷综合久久中文字幕蜜桃三电影| 国产精品女同一区二区久久| 久久亚洲精品人成综合网| 久久综合色区| 精品久久久久国产免费 | 亚洲国产成人乱码精品女人久久久不卡| 亚洲精品无码久久千人斩| 亚洲国产精品嫩草影院久久| 97久久超碰成人精品网站| 亚洲精品无码久久久久久| 亚洲Av无码国产情品久久| 久久播电影网| 伊人久久大香线蕉精品| 99久久国产热无码精品免费| 伊人久久大香线蕉AV色婷婷色| 久久久久综合国产欧美一区二区 | 一本久久a久久精品综合夜夜| 91精品国产9l久久久久| 2021精品国产综合久久| 久久91亚洲人成电影网站| 久久久青草青青亚洲国产免观| 国产精品久久亚洲不卡动漫|