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

洛譯小筑

別來無恙,我的老友…
隨筆 - 45, 文章 - 0, 評論 - 172, 引用 - 0
數據加載中……

[ECPP讀書筆記 條目8] 防止因異常中止析構函數

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

class Widget {

public:

  ...

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

};

 

void doSomething()

{

  std::vector<Widget> v;

  ...

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

vector v被銷毀時,它也有責任銷毀其所包含的所有的Widget。假設v中包含十個Widget,并且在對第一個進行析構時拋出了一個異常。那么剩下的九個Widget則仍需要得到銷毀(否則它們所占有的資源就會發生泄漏),所以v應該為所有剩下的Widget——調用析構函數。但是假設在對這些對象進行銷毀時,又出現了第二個Widget拋出了一個異常,現在同時存在著兩個活動的異常,這對于C++來說已經是太多了。在極端巧合的情形下,程序中同時出現了兩個活動的異常,此時程序的運行要么會中止,要么會產生未定義行為。本示例將產生未定義行為。在使用其它的標準庫容器(比如listset等),任意的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函數,這樣就為客戶提供了途徑來處理由DBConnectionclose產生的異常。這樣做還可以跟蹤DBConnection所建立的連接是否被DBConnection自己的close函數正常關閉,如果關閉失敗則在DBConn的析構函數將其關閉。這可以防止已建立的連接發生泄漏。然而,如果在DBConn的析構函數中對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會忽略這個異常或者終止程序。客戶對此也沒有什么好抱怨的,畢竟,在處理問題時是他們犯下了第一個錯誤,是他們自己選擇不去利用它的。

時刻牢記

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

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

posted on 2007-04-22 14:00 ★ROY★ 閱讀(1466) 評論(3)  編輯 收藏 引用 所屬分類: Effective C++

評論

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構函數)  回復  更多評論   

我支持你!
2007-04-27 10:06 | sandy

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構函數)  回復  更多評論   

謝謝啊,這兩天裝系統總出問題,腦袋都裝大了。。
2007-04-27 10:51 | ★田德健★

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構函數)  回復  更多評論   

千萬不要沖動~裝系統是最讓我郁悶的事~我曾經深受其害!!
2007-05-05 21:54 | sandy
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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精品欧美一区二区蜜桃免费| 日韩视频在线一区二区| 国内精品久久久久久久果冻传媒 | 男人插女人欧美| 久久久国产91| 美女脱光内衣内裤视频久久影院 | 欧美在线视频二区| 久久国产成人| 美女亚洲精品| 欧美日韩国产成人在线| 国产精品久久二区二区| 国产一区二区主播在线| 亚洲精品乱码| 欧美一二三区在线观看| 老牛影视一区二区三区| 亚洲国产网站| 蜜桃av久久久亚洲精品| 亚洲精品日韩综合观看成人91| 亚洲裸体视频| 亚洲免费网址| 欧美α欧美αv大片| 国产精品卡一卡二| 伊人久久男人天堂| 在线免费观看日本一区| 欧美精品色一区二区三区| 国产精品成人观看视频免费| 国产情人综合久久777777| 亚洲成色777777女色窝| 亚洲欧美日本伦理| 亚洲国产精品v| 亚洲综合国产激情另类一区| 免费观看日韩| 国产亚洲va综合人人澡精品| 亚洲乱码国产乱码精品精 | 国产午夜精品视频免费不卡69堂| 亚洲电影免费观看高清完整版| 在线视频免费在线观看一区二区| 久久视频精品在线| 亚洲视频大全| 欧美精品一区二区精品网| 国产婷婷色一区二区三区| 日韩午夜av在线| 免费成人性网站| 香蕉久久夜色精品| 国产精品二区二区三区| 亚洲另类一区二区| 欧美成人精品在线视频| 午夜精彩视频在线观看不卡| 欧美日韩免费| 一区二区三区高清在线| 亚洲国产精品传媒在线观看 | 美女成人午夜| 国内久久婷婷综合| 久久精品国产免费观看| 亚洲欧美日韩成人| 国产精品美女一区二区在线观看| 在线一区欧美| 亚洲免费观看高清完整版在线观看熊 | 亚洲一区二区三区色| 亚洲丰满在线| 免费看黄裸体一级大秀欧美| 黄色亚洲免费| 牛夜精品久久久久久久99黑人| 欧美伊人久久| 精品99一区二区三区| 另类天堂av| 久久久中精品2020中文| 伊人色综合久久天天| 蜜桃av久久久亚洲精品| 免费看成人av| 中文av字幕一区| 亚洲视频播放| 狠狠狠色丁香婷婷综合久久五月 | 国产一区视频观看| 国产精品久久久久77777| 亚洲图中文字幕| 99精品99久久久久久宅男| 欧美日韩亚洲一区| 亚洲欧美日韩第一区| 亚洲香蕉网站| 国产日韩精品入口| 久久人人爽人人爽爽久久| 久久午夜电影网| 亚洲精品欧美激情| 一区二区日韩伦理片| 国产麻豆综合| 免费久久精品视频| 欧美精品xxxxbbbb| 午夜精品一区二区三区在线视| 亚洲欧美国产高清va在线播| 狠狠色狠狠色综合人人| 亚洲激情国产精品| 国产精品色婷婷| 免费人成精品欧美精品| 欧美日韩一区二区在线观看 | 亚洲欧美一区二区三区久久| 亚洲欧美制服另类日韩| 亚洲国产成人tv| 中国女人久久久| 在线高清一区| 亚洲一区二区三区涩| 亚洲第一搞黄网站| 亚洲尤物在线| 亚洲毛片网站| 欧美在线观看视频一区二区三区 | 精久久久久久久久久久| 亚洲精品精选| 国产在线观看91精品一区| 最近看过的日韩成人| 国产日韩欧美中文在线播放| 亚洲成人在线视频播放 | 久久午夜激情| 欧美私人网站| 亚洲高清在线观看| 国内外成人免费激情在线视频网站| 亚洲人成网在线播放| 国产有码在线一区二区视频| 亚洲免费成人| 亚洲看片免费| 噜噜噜噜噜久久久久久91| 久久爱91午夜羞羞| 国产精品国产三级国产aⅴ入口 | 亚洲欧美美女| 欧美精品电影| 亚洲天堂成人| 欧美激情视频一区二区三区免费 | 亚洲少妇一区| 欧美成人性生活| 免费高清在线一区| 精品动漫一区| 欧美中文字幕精品| 欧美一区二区三区四区夜夜大片| 欧美女激情福利| 欧美激情一区| 亚洲欧洲综合另类| 久久亚洲综合色| 蜜臀av在线播放一区二区三区| 国产午夜亚洲精品羞羞网站 | 亚洲啪啪91| 99国产精品国产精品久久| 免费成人av在线看| 亚洲成人资源| 一区二区三区 在线观看视| 欧美精品一区二区高清在线观看| 亚洲电影在线| 99人久久精品视频最新地址| 欧美成人嫩草网站| 亚洲人永久免费| 亚洲午夜电影在线观看| 欧美偷拍一区二区| 亚洲欧美日韩精品综合在线观看| 午夜精品www| 国外成人在线视频| 久久综合色8888| 亚洲七七久久综合桃花剧情介绍| 一区二区三区视频在线 | 噜噜噜在线观看免费视频日韩| 亚洲成色精品| 亚洲性感美女99在线| 国产精品亚洲综合久久| 欧美一级黄色网| 免播放器亚洲一区| 久久综合伊人| 国产日韩精品视频一区| 红桃av永久久久| 久久国产精品色婷婷| 久久精品色图| 亚洲精品乱码久久久久| 99国产精品99久久久久久粉嫩| 国产精品v欧美精品v日韩| 先锋资源久久| 欧美91精品| 欧美亚洲免费高清在线观看| 先锋影音国产精品| 亚洲日本国产| 性欧美xxxx大乳国产app| 亚洲国产小视频| 中文一区二区| 久久国产天堂福利天堂| 国产伦精品一区二区三区照片91 | 久久精品国产第一区二区三区最新章节| 国产精品日日摸夜夜添夜夜av| 久久久免费观看视频| 亚洲美女黄网| 美女日韩在线中文字幕| 亚洲欧美国产va在线影院| 亚洲成人在线视频播放| 国产精品国产三级国产普通话蜜臀| 久久精品国产第一区二区三区最新章节| 欧美国产日韩一区二区三区| 亚洲欧美在线另类| 亚洲精品免费网站| 狠狠综合久久av一区二区小说| 欧美手机在线视频| 欧美成在线视频| 久久久久久久欧美精品| 午夜在线视频观看日韩17c| 99精品国产福利在线观看免费| 欧美成人一品| 狂野欧美激情性xxxx|