青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Shuffy

不斷的學習,不斷的思考,才能不斷的進步.Let's do better together!
posts - 102, comments - 43, trackbacks - 0, articles - 19

[轉]http://m.shnenglu.com/tiandejian/archive/2007/04/22/ecpp_08.html
第8條:
防止因異常而中止析構函數

C++ 沒有禁止析構函數引發異常,但 C++ 分不推薦這一做法。這樣做有充足的理由。請看下邊的代碼:

class Widget {

public:

 ...

 ~Widget() { ... }                    // 假設它會引發一個異常

};

 

void doSomething()

{

 std::vector<Widget> v;

 ...

}                                      // v 在這里被自動銷毀

vector v 被銷毀時,它也有責任銷毀其所包含的所有的 Widget 。假設 v 中包含十個 Widget ,并且在對第一個進行 析構時拋出了一個異常。那么剩下的九 Widget 則仍需要得到銷毀(否則它們所占有的資源就會發生泄漏),所以 v 應該為所有剩下的 Widget 一一調用析構函數。但是假設在對這些對象進行銷毀時,又出現了第二個 Widget 出了一個異常,現在同時存在著兩個活動的異常,這對 C++ 來說 已經是太多了。在極端巧合的情形下,程序中同時出現了兩個活動的異常,此時程序的運行要么會中止,要么會產生無法預知的行為。本示例將產生無法預知的行為。在使用其它的標準庫容器(比 list set 等),任意的 TR1 器(參見 54 ),甚至是一個數組,同樣都會產生無法預知的行為。然而為你帶來麻煩的不僅僅是這些容器或者數組,析構函數拋出異常會引發不成熟的程序終止或者未知的行為,甚至在沒有容器和數組的情況下也會發生。 C++ 不喜歡能夠引發異常的析構函數!

這個問題很好理解,但是當你的析構函數的某一操作可能失敗,并且有可能對這一失敗的操作拋出一個異常時,你應該怎么做呢?請看下邊的示例,其中假設你使用一個類進行數據庫連接:

class DBConnection {

public:

 ...

 static DBConnection create();  // 返回 DBConnection 對象的函數;

                                  // 為簡化代碼省略了參數表

 

 void close();                   // 關閉連接;若關閉失敗則拋出異常

};

為了確保客戶端程序員不會忘記 DBConnection 對象調 close 函數,一個可行的方案是:創建一個新的類來管理 DBConnection 的資源,在這個類的析構函數中調用 close 。這種資源管理類在第三章中作詳細的介紹,在本節中,我們僅關心這些類的析構函數是什么樣的:

class DBConn {                // 該類用來管理 DBConnection 對象的資源

public:

 ...

 ~DBConn()                    // 確保數據庫連接總能關閉

   {

   db.close();

 }

private:

 DBConnection db;

};

客戶端程序員可以這樣編寫:

{                              // 開始一個程序塊

   DBConn dbc(DBConnection::create());

                               // 創建一個 DBConnection 對象,然后

                               // 把它交給一個 DBConn 對象來管理

 

...                            // 通過 DBConn 的接口使用這個

                               // DBConnection 對象

 

}                              // 在該程序塊的最后,這個 DBConn 對象

                               // 被銷毀了,就好像自動調用了那個

                               // DBConnection 對象的 close 函數

只要對 close 的調用能成功,這個 DBConn 的方案就是一個好主意,但是一旦這一調用會引發一個異常, DBConn 的析構函數則會使這個異常蔓延開來,也就是所謂的,允許“因異常而中止析構函數”。這便是問題所在,因為析構函數在此處拋出異常意味著麻煩將會出現。

避免這類麻煩有兩種主要的辦法。 DBConn 的析構函數可以:

如果 close 拋出異常則終止程序,通常通過調用 abort 實現

DBConn::~DBConn()

{

 try { db.close(); }

 catch (...) {

   在日志上記載:調用 close 失敗 ;

   std::abort();

 }

}

如果在析構過程中發生了一個錯誤,從而程序無法運行下去了,上面的方法就是一個可行的選擇。如果允許析構函數傳播異常將導致程序行為無法預知,這樣做的優勢就在于可以避免類似的事情發生。也就是說,調 abort 數可以 防止程序產生未知的行為。

忽略這個異常 —— 由調用 close 函數產生的異常

DBConn::~DBConn()

{

   try { db.close(); }

   catch (...) {

      在日志上記載:調用 close 失敗 ;

 }

}

在大多數情況下,忽略掉異常的存在并不是一個好主意,因為這樣做你會錯過一些重要的信息——一些東西出錯了!然而在某些時刻,忽略異常比讓程序擔上不成熟終止或未知行為的風險要強一些。為了讓忽略異常成為一個可行的方案,程序必須有能力在發生的錯誤被忽略之后仍然可以穩定地繼續運行。

這兩個方案都不是那么的動人心弦。這兩者存在著同樣的問題,它們沒有辦法在第一時間對 close 拋出的異常作出反應。

一個更好的策略是:改進 DBConn 接口的設計,使得客戶端程序員有機會自己處理可能發生的問題。舉例說, DBConn 可以自己包含一個 close 函數,這樣就為客戶端程序員提供了途徑來處理由 DBConnection close 產生的異常。這樣做還可以保持跟蹤 DBConnection 所建立的連接是否被 DBConnection 自己的 close 函數正常關閉,如果關閉失敗則在 DBConn 的析構函數再次嘗試。這可以防止已建立的連接發生泄漏。然而,如果在 DBConn 的析構函數 [1] 中對 close 的調用仍然不成功,我們還是需要中止運行或者忽略異常。

class DBConn {

public:

 ...

 void close()                        // 新函數,供客戶端程序員調用

 {

    db.close();

    closed = true;

 }

 

 ~DBConn()

   {

  if (!closed) {

   try {                              // 如果客戶端程序員沒有關閉連接,

      db.close();                      // 則在這里關閉它

   }

   catch (...) {                      // 如果沒有正常關閉,

      在日志上記載:調用 close 失敗 ;      // 首先作好記錄,

     ...                              // 然后終止或忽略

  }

 }

 

private:

 DBConnection db;

   bool closed;

};

調用 close 的責任原本是 DBConn 的析構函數的,而現在我們卻將其轉交給 DBConn 的客戶端程序員( DBConn 的析構函數還包含一個“備用的”調用)。可能你會認為這樣做實屬毫無顧忌地推卸責任,你甚至可能認為這是對“讓接口更簡單易用”這一忠告(見第 18 條)的違背。實際上,兩者都不是。如果一個操作可能由于一次異常的拋出而失敗,同時這個異常有需要得到處理,這一異常不應該來自析構函數。這是因為引發異常的析構函數是十分危險的,它使你的程序始終位于風口浪尖:你無法避免不成熟的終止和未知行為。在上邊的示例中,讓客戶端程序員自己手動調用 close 并不會為其帶來過多的負擔,相反地,這樣做為客戶端程序員提供了處理錯誤的機會,否則他們就會束手無策。如果他們沒有發現這個機會的裨益所在(可能是因為他們相信錯誤不會發生得這么巧),他們可以忽略這個機會,然后依賴 DBConn 的析構函數為他們調用 close 函數。如果就在這一刻發生了錯誤——也就是說 close 確實拋出了異常—— DBConn 會忽略這個異常或者終止程序。客戶端程序員對此也沒有什么好抱怨的,畢竟,在處理問題時是他們犯下了第一個錯誤,是他們自己選擇不去利用它的。

需要記住的

永遠不要讓析構函數引發異常。如果析構函數所調用的函數會拋出異常的話,那么析構函數中要捕捉到所有異常,然后忽略它們或者終止程序。

在一次操作中,如果一個類的使用者有能力對拋出異常作出反應,那么這個類應該提供一個常規的函數(而不是析構函數)來進行這一操作。



[1] 原文有 誤( DBConnection 的析構函數),此處始終未涉及到 DBConnection 的析 構函數。——譯注

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品啊啊啊| 欧美日韩精品免费观看视频| 欧美在线视频网站| 欧美日本免费| 在线观看视频一区| 久久精品人人做人人爽| 99热精品在线| 欧美剧在线免费观看网站| 尤物精品国产第一福利三区| 欧美一区二区三区四区在线| 宅男噜噜噜66一区二区 | 亚洲免费成人| 麻豆成人综合网| 欧美呦呦网站| 国产亚洲欧美激情| 久久久久久久久综合| 亚洲欧美日韩另类| 国产一区白浆| 蜜臀av一级做a爰片久久| 久久人体大胆视频| 亚洲国产另类久久久精品极度| 欧美成人亚洲| 欧美精品一区二区在线播放| 亚洲美女福利视频网站| 亚洲精品一区二区三区在线观看| 欧美日韩激情网| 午夜欧美理论片| 久久欧美中文字幕| 亚洲欧洲一区二区三区| 91久久在线播放| 欧美丝袜一区二区| 性做久久久久久久免费看| 午夜精品久久久久久久蜜桃app| 国产日韩欧美a| 美女性感视频久久久| 久久亚洲高清| 亚洲天堂成人在线观看| 亚洲主播在线| 在线观看国产成人av片| 亚洲激情亚洲| 国产精品一区免费视频| 久久―日本道色综合久久| 久久亚洲综合色| 亚洲欧美日本另类| 久久精品男女| 一本到12不卡视频在线dvd| 亚洲女爱视频在线| 亚洲电影免费观看高清完整版在线| 影音先锋国产精品| 欧美激情视频在线播放| 欧美午夜一区二区福利视频| 欧美有码视频| 一本在线高清不卡dvd| 国产一区二区三区四区在线观看| 欧美国产精品va在线观看| 欧美视频久久| 欧美激情中文字幕乱码免费| 国产精品区一区| 亚洲国产精品尤物yw在线观看| 国产精品久久久久久久久免费樱桃| 久久偷看各类wc女厕嘘嘘偷窃| 欧美国产一区二区在线观看| 久久婷婷人人澡人人喊人人爽| 欧美日韩中文字幕在线| 免费亚洲网站| 久久久中精品2020中文| 亚洲欧美日韩精品久久亚洲区| 久久青草欧美一区二区三区| 亚洲欧美日韩区| 欧美国产日韩亚洲一区| 久久精品中文字幕免费mv| 欧美激情综合色综合啪啪| 久久亚洲精品一区| 亚洲欧美精品一区| 亚洲免费激情| 六月婷婷一区| 久久五月婷婷丁香社区| 国产精品福利影院| 亚洲人成人77777线观看| 激情久久影院| 亚洲欧美三级伦理| 亚洲一区在线播放| 欧美日韩不卡合集视频| 久久国产88| 久久精选视频| 一本久久综合亚洲鲁鲁五月天| 韩国三级电影久久久久久| 亚洲视频一二三| 一区二区不卡在线视频 午夜欧美不卡在 | 亚洲日本视频| 国产亚洲午夜| 在线不卡视频| 欧美激情视频网站| 玖玖精品视频| 日韩午夜在线视频| 香蕉久久国产| 亚洲欧洲日产国码二区| 亚洲国产日韩一区二区| 欧美一区在线直播| 亚洲精品免费在线播放| 亚洲激情视频网| 欧美日韩专区在线| 亚洲第一中文字幕在线观看| 欧美日韩免费看| 性做久久久久久久久| 欧美精品在线极品| 午夜亚洲影视| 亚洲女女做受ⅹxx高潮| 久久综合给合| 免费国产一区二区| 欧美一区二区在线看| 一区二区三区偷拍| 亚洲成色999久久网站| 亚洲日本中文字幕免费在线不卡| 欧美日韩国产在线播放| 欧美一二三区在线观看| 欧美一区二区三区久久精品| 一个色综合av| 暖暖成人免费视频| 亚洲先锋成人| 国产视频一区在线观看| 久久青草久久| 免费日韩视频| 亚洲国产综合91精品麻豆| 国产精品毛片a∨一区二区三区|国| 久久九九精品99国产精品| 美女日韩在线中文字幕| 亚洲欧美日韩精品久久奇米色影视| 亚洲精品永久免费精品| 在线观看av不卡| 亚洲精品小视频| 亚洲精品综合| 欧美日韩精品在线播放| 欧美高清在线一区二区| 国产欧美日韩亚洲一区二区三区| 欧美在线视频观看免费网站| 亚洲五月六月| 欧美极品在线视频| 久久久久久久尹人综合网亚洲| 国产精品免费一区二区三区在线观看| 亚洲精品一区二区三区樱花| 国产欧美日韩视频在线观看| 一区二区电影免费在线观看| 亚洲一区二区影院| 久久久精品国产免大香伊| 日韩视频一区二区| 老巨人导航500精品| 亚洲一区二区三区免费观看| 国产欧美综合一区二区三区| 久久久久久自在自线| 亚洲午夜免费福利视频| 久久精品在线视频| 在线播放中文一区| 欧美黄色成人网| 亚洲精品国产系列| 亚洲欧美日韩天堂| 国产日韩欧美亚洲一区| 欧美精品久久久久久久久老牛影院 | 国产精品欧美日韩一区| 中文无字幕一区二区三区| 亚洲综合丁香| 最新国产成人av网站网址麻豆| 麻豆成人精品| 欧美777四色影视在线| 一区二区成人精品 | 欧美va天堂在线| 亚洲婷婷综合色高清在线 | 亚洲高清免费| 欧美日韩在线一区| 欧美亚洲日本网站| 亚洲伦理在线| 一区二区三区免费在线观看| 欧美一区二区三区在线免费观看| 极品尤物av久久免费看| 国产精品天美传媒入口| 欧美大片91| 久久国产精品久久久久久久久久 | 国产精品海角社区在线观看| 欧美gay视频| 99伊人成综合| 在线观看日韩欧美| 国产精品久久久久久久久久妞妞 | 国产视频欧美视频| 亚洲精品欧美极品| 亚洲一区二区在线播放| 国产精品亚洲综合色区韩国| 欧美日韩视频一区二区| 欧美亚洲免费电影| 欧美专区日韩视频| 一区二区福利| 亚洲精品国产欧美| 亚洲欧洲日本一区二区三区| 欧美一区1区三区3区公司| 亚洲私人影院在线观看| 亚洲精选久久| 亚洲精品欧美日韩| 国产午夜精品全部视频播放 | 欧美性jizz18性欧美| 午夜在线观看欧美| 美女精品在线|