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

洛譯小筑

別來(lái)無(wú)恙,我的老友…
隨筆 - 45, 文章 - 0, 評(píng)論 - 172, 引用 - 0
數(shù)據(jù)加載中……

[ECPP讀書(shū)筆記 條目8] 防止因異常中止析構(gòu)函數(shù)

C++并沒(méi)有禁止析構(gòu)函數(shù)引發(fā)異常,但是C++無(wú)疑不會(huì)推薦這一做法。這樣做有充足的理由。請(qǐng)看下邊的代碼:

class Widget {

public:

  ...

  ~Widget() { ... }                // 假設(shè)它會(huì)引發(fā)一個(gè)異常

};

 

void doSomething()

{

  std::vector<Widget> v;

  ...

}                                  // v在這里被自動(dòng)銷毀

當(dāng)vector v被銷毀時(shí),它也有責(zé)任銷毀其所包含的所有的Widget。假設(shè)v中包含十個(gè)Widget,并且在對(duì)第一個(gè)進(jìn)行析構(gòu)時(shí)拋出了一個(gè)異常。那么剩下的九個(gè)Widget則仍需要得到銷毀(否則它們所占有的資源就會(huì)發(fā)生泄漏),所以v應(yīng)該為所有剩下的Widget——調(diào)用析構(gòu)函數(shù)。但是假設(shè)在對(duì)這些對(duì)象進(jìn)行銷毀時(shí),又出現(xiàn)了第二個(gè)Widget拋出了一個(gè)異常,現(xiàn)在同時(shí)存在著兩個(gè)活動(dòng)的異常,這對(duì)于C++來(lái)說(shuō)已經(jīng)是太多了。在極端巧合的情形下,程序中同時(shí)出現(xiàn)了兩個(gè)活動(dòng)的異常,此時(shí)程序的運(yùn)行要么會(huì)中止,要么會(huì)產(chǎn)生未定義行為。本示例將產(chǎn)生未定義行為。在使用其它的標(biāo)準(zhǔn)庫(kù)容器(比如listset等),任意的TR1容器(參見(jiàn)條目54),甚至是一個(gè)數(shù)組,同樣都會(huì)產(chǎn)生未定義行為。然而為你帶來(lái)麻煩的不僅僅是這些容器或者數(shù)組,析構(gòu)函數(shù)拋出異常會(huì)引發(fā)不成熟的程序終止或者未定義行為,甚至在沒(méi)有容器和數(shù)組的情況下也會(huì)發(fā)生。C++不喜歡能夠引發(fā)異常的析構(gòu)函數(shù)!

這個(gè)問(wèn)題很好理解,但是當(dāng)你的析構(gòu)函數(shù)的某一操作可能失敗,并且有可能拋出一個(gè)異常時(shí),你應(yīng)該怎么做呢?請(qǐng)看下邊的示例,其中假設(shè)你使用一個(gè)類進(jìn)行數(shù)據(jù)庫(kù)連接:

class DBConnection {

public:

  ...

  static DBConnection create();    // 返回DBConnection對(duì)象的函數(shù);

                                   // 為簡(jiǎn)化代碼省略了參數(shù)表

 

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

};

為了確保客戶不會(huì)忘記為DBConnection對(duì)象調(diào)用close函數(shù),一個(gè)可行的方案是:創(chuàng)建一個(gè)新的類來(lái)管理DBConnection的資源,在這個(gè)類的析構(gòu)函數(shù)中調(diào)用close。這種資源管理類在第三章中作詳細(xì)的介紹,在本節(jié)中,我們僅關(guān)心這些類的析構(gòu)函數(shù)是什么樣的:

class DBConn {                     // 該類用來(lái)管理 DBConnection對(duì)象的資源

public:

  ...

  ~DBConn()                        // 確保數(shù)據(jù)庫(kù)連接總能關(guān)閉

  {

   db.close();

  }

private:

  DBConnection db;

};

客戶可以這樣編寫(xiě):

{                                  // 開(kāi)始一個(gè)程序塊

   DBConn dbc(DBConnection::create());

                                   // 創(chuàng)建一個(gè) DBConnection對(duì)象,然后

                                   // 把它交給一個(gè) DBConn對(duì)象來(lái)管理

 

...                                // 通過(guò)DBConn的接口使用這個(gè)

                                   // DBConnection對(duì)象

 

}                                  // 在該程序塊的最后,這個(gè)DBConn對(duì)象

                                   // 被銷毀了,就好像自動(dòng)調(diào)用了那個(gè)

                                   // DBConnection對(duì)象的close函數(shù)

只要對(duì)close的調(diào)用能成功,這個(gè)DBConn的方案就是一個(gè)好主意,但是一旦這一調(diào)用會(huì)引發(fā)一個(gè)異常, DBConn的析構(gòu)函數(shù)則會(huì)使這個(gè)異常蔓延開(kāi)來(lái),也就是所謂的,允許“因異常而中止析構(gòu)函數(shù)”。這便是問(wèn)題所在,因?yàn)槲鰳?gòu)函數(shù)在此處拋出異常意味著麻煩將會(huì)出現(xiàn)。

避免這類麻煩有兩種主要的辦法。DBConn的析構(gòu)函數(shù)可以:

終止程序——如果close拋出異常則終止程序通常通過(guò)調(diào)用abort實(shí)現(xiàn):

DBConn::~DBConn()

{

 try { db.close(); }

 catch (...) {

   在日志上記載:調(diào)用close失敗;

   std::abort();

 }

}

如果在析構(gòu)過(guò)程中發(fā)生了一個(gè)錯(cuò)誤,從而程序無(wú)法繼續(xù)運(yùn)行,上面的方法就是一個(gè)可行的選擇。如果允許析構(gòu)函數(shù)傳播異常將導(dǎo)致程序產(chǎn)生未定義行為,這樣做的優(yōu)勢(shì)就在于可以避免類似的事情發(fā)生。也就是說(shuō),調(diào)用abort函數(shù)可以防止程序產(chǎn)生未定義行為。

忽略這個(gè)異常——由調(diào)用close函數(shù)產(chǎn)生的異常

DBConn::~DBConn()

{

  try { db.close(); }

  catch (...) {

      在日志上記載:調(diào)用close失敗;

  }

}

在大多數(shù)情況下,忽略掉異常的存在并不是一個(gè)好主意,因?yàn)檫@樣做你會(huì)錯(cuò)過(guò)一些重要的信息——一些東西出錯(cuò)了!然而在某些時(shí)刻,忽略異常比讓程序擔(dān)上不成熟終止或未定義行為的風(fēng)險(xiǎn)要強(qiáng)一些。為了讓忽略異常成為一個(gè)可行的方案,程序必須有能力在忽略剛發(fā)生的錯(cuò)誤之后仍然可以穩(wěn)定地繼續(xù)運(yùn)行。

這兩個(gè)方案都不是那么的動(dòng)人心弦。它們存在著同樣的問(wèn)題,即二者都沒(méi)有辦法在第一時(shí)間對(duì)close拋出的異常作出反應(yīng)。

一個(gè)更好的策略是:改進(jìn)DBConn接口的設(shè)計(jì),使得客戶有機(jī)會(huì)自己處理可能發(fā)生的問(wèn)題。舉例說(shuō),DBConn可以自己提供一個(gè)close函數(shù),這樣就為客戶提供了途徑來(lái)處理由DBConnectionclose產(chǎn)生的異常。這樣做還可以跟蹤DBConnection所建立的連接是否被DBConnection自己的close函數(shù)正常關(guān)閉,如果關(guān)閉失敗則在DBConn的析構(gòu)函數(shù)將其關(guān)閉。這可以防止已建立的連接發(fā)生泄漏。然而,如果在DBConn的析構(gòu)函數(shù)中對(duì)close的調(diào)用仍然不成功,我們還是需要中止運(yùn)行或者忽略異常。

class DBConn {

public:

  ...

  void close()                     // 新函數(shù),供客戶調(diào)用

  {

    db.close();

    closed = true;

  }

 

  ~DBConn()

   {

   if (!closed) {

   try {                           // 如果客戶沒(méi)有關(guān)閉連接,則在這里關(guān)閉它

     db.close();

   }

   catch (...) {                   // 如果沒(méi)有正常關(guān)閉,首先作好記錄,

     在日志上記載:調(diào)用close失敗;     // 然后終止或忽略

     ...

   }

  }

 

private:

  DBConnection db;

  bool closed;

};

調(diào)用close的責(zé)任原本是DBConn的析構(gòu)函數(shù)的,而現(xiàn)在我們卻將其轉(zhuǎn)交給DBConn的客戶(DBConn的析構(gòu)函數(shù)還包含一個(gè)“備用的”調(diào)用)。可能你會(huì)認(rèn)為這樣做實(shí)屬毫無(wú)顧忌地推卸責(zé)任,你甚至可能認(rèn)為這是對(duì)“讓接口更易于正確用”這一忠告(見(jiàn)條目18)的違背。實(shí)際上,兩者都不是。如果一個(gè)操作可能由于一次異常的拋出而失敗,同時(shí)這個(gè)異常有必要得到處理,這一異常不應(yīng)該來(lái)自析構(gòu)函數(shù)。這是因?yàn)橐l(fā)異常的析構(gòu)函數(shù)是十分危險(xiǎn)的,它使你的程序始終位于風(fēng)口浪尖:你無(wú)法避免不成熟的終止和未定義行為的風(fēng)險(xiǎn)。在上邊的示例中,讓客戶自己手動(dòng)調(diào)用close并不會(huì)為其帶來(lái)過(guò)多的負(fù)擔(dān),相反地,這樣做為客戶提供了處理那些原本他們無(wú)法接觸的錯(cuò)誤的機(jī)會(huì)。如果他們沒(méi)有發(fā)現(xiàn)這個(gè)機(jī)會(huì)的裨益所在(可能是因?yàn)樗麄兿嘈佩e(cuò)誤不會(huì)發(fā)生得這么巧),他們可以忽略這個(gè)機(jī)會(huì),然后依賴DBConn的析構(gòu)函數(shù)為他們調(diào)用close函數(shù)。如果就在這一刻發(fā)生了錯(cuò)誤——也就是說(shuō)close確實(shí)拋出了異常——DBConn會(huì)忽略這個(gè)異常或者終止程序。客戶對(duì)此也沒(méi)有什么好抱怨的,畢竟,在處理問(wèn)題時(shí)是他們犯下了第一個(gè)錯(cuò)誤,是他們自己選擇不去利用它的。

時(shí)刻牢記

永遠(yuǎn)不要讓析構(gòu)函數(shù)引發(fā)異常。如果析構(gòu)函數(shù)所調(diào)用的函數(shù)會(huì)拋出異常的話,那么析構(gòu)函數(shù)中要捕捉到所有異常,然后忽略它們或者終止程序。

在一次操作中,如果類的客戶有必要對(duì)拋出的異常做出反應(yīng),那么這個(gè)類應(yīng)該提供一個(gè)常規(guī)的函數(shù)(而不是析構(gòu)函數(shù))來(lái)進(jìn)行這一操作。

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

評(píng)論

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構(gòu)函數(shù))  回復(fù)  更多評(píng)論   

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

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構(gòu)函數(shù))  回復(fù)  更多評(píng)論   

謝謝啊,這兩天裝系統(tǒng)總出問(wèn)題,腦袋都裝大了。。
2007-04-27 10:51 | ★田德健★

# re: 【翻譯】Effective C++ (第8條:防止因異常而中止析構(gòu)函數(shù))  回復(fù)  更多評(píng)論   

千萬(wàn)不要沖動(dòng)~裝系統(tǒng)是最讓我郁悶的事~我曾經(jīng)深受其害!!
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>
            在线亚洲激情| 久久综合九色九九| 久久久蜜桃一区二区人| 亚洲精品国产精品国自产在线 | 欧美暴力喷水在线| 久久青青草原一区二区| 91久久精品www人人做人人爽| 午夜精品久久久久| 午夜在线a亚洲v天堂网2018| 欧美精品在线免费| 欧美人成在线视频| 新狼窝色av性久久久久久| 久久久久免费视频| 久久久久久尹人网香蕉| 91久久久在线| 亚洲一区二区三区777| 欧美日韩在线亚洲一区蜜芽| 亚洲深爱激情| 亚洲精品欧美精品| 亚洲电影av| 国产精品啊v在线| 久久亚洲春色中文字幕| 亚洲人成精品久久久久| 亚洲午夜精品一区二区三区他趣| 午夜影院日韩| 伊甸园精品99久久久久久| 西瓜成人精品人成网站| 欧美一级一区| 亚洲另类在线视频| 国产精品第13页| 亚洲一区二区三区色| 久久精品在线播放| 亚洲国产精品久久久久秋霞不卡 | 欧美性猛交视频| 亚洲自拍都市欧美小说| 亚洲影视综合| 一区福利视频| 亚洲国产欧美国产综合一区| 欧美激情一区二区三区在线视频| 9国产精品视频| 亚洲免费在线看| 伊人夜夜躁av伊人久久| 亚洲日本欧美| 欧美一区二区啪啪| 欧美精品在线观看91| 亚洲欧美日韩精品久久奇米色影视 | 亚洲欧美成aⅴ人在线观看| 午夜亚洲影视| 一本大道久久a久久精二百| 亚洲欧美另类国产| 亚洲国产小视频在线观看| 亚洲深爱激情| 亚洲精品一线二线三线无人区| 9久re热视频在线精品| 激情综合色综合久久| 99av国产精品欲麻豆| 狠狠色2019综合网| 在线亚洲自拍| 亚洲狼人综合| 久久裸体视频| 欧美综合国产| 国产精品videossex久久发布| 男人的天堂成人在线| 国产精品日韩电影| 亚洲精品中文字幕女同| 在线成人小视频| 亚洲欧美日韩综合| 亚洲一区久久久| 免费看精品久久片| 久久久久久久999精品视频| 欧美韩国日本一区| 欧美激情欧美狂野欧美精品 | 午夜精品国产更新| 欧美精品久久久久久| 麻豆成人av| 国产无遮挡一区二区三区毛片日本| 亚洲精品乱码久久久久久日本蜜臀 | 一本一本久久| 日韩视频中午一区| 美女视频黄免费的久久| 久久九九全国免费精品观看| 国产精品激情av在线播放| 亚洲三级电影在线观看| 亚洲国产裸拍裸体视频在线观看乱了中文| 亚洲一区二区在线播放| 宅男在线国产精品| 欧美女同视频| 亚洲免费成人av| 一区二区三区欧美在线观看| 欧美人妖另类| 亚洲美女毛片| 亚洲一区日韩在线| 国产精品不卡在线| 亚洲视频狠狠| 欧美一级在线播放| 国产日韩欧美在线播放| 欧美一级久久| 久久久久久久网站| 激情综合网址| 美腿丝袜亚洲色图| 亚洲第一黄色| 一本久久a久久精品亚洲| 欧美巨乳在线观看| 久久人人97超碰精品888| 亚洲一区久久| 狠狠色噜噜狠狠色综合久| 欧美午夜无遮挡| 欧美日韩国产高清视频| 国产视频精品xxxx| 亚洲片区在线| 欧美专区日韩专区| 欧美大尺度在线观看| 99伊人成综合| 蜜桃av久久久亚洲精品| 香蕉av777xxx色综合一区| 午夜精品久久久久久久99黑人| 亚洲欧美一区二区激情| 欧美电影免费观看大全| 黄色成人91| 久久久另类综合| 日韩亚洲不卡在线| 欧美岛国激情| 亚洲精品一二| 亚洲乱码一区二区| 欧美国产一区二区在线观看| 好吊妞**欧美| 久久人人精品| 久久精品欧美| 91久久国产自产拍夜夜嗨| 免费成人av资源网| 久久综合久久久久88| 亚洲国产精品va在线看黑人| 欧美mv日韩mv国产网站| 久久躁日日躁aaaaxxxx| 在线观看视频欧美| 欧美黄网免费在线观看| 免费亚洲婷婷| 亚洲欧美日韩专区| 亚洲一级特黄| 欧美乱妇高清无乱码| 亚洲视频在线免费观看| 亚洲少妇在线| 亚洲国产成人av| 亚洲国产天堂久久综合网| 91久久久久| 欧美调教视频| 久久亚洲精品视频| 欧美激情一区在线| 性做久久久久久久久| 久久久夜夜夜| 香蕉成人啪国产精品视频综合网| 久久精品一区四区| 91久久国产综合久久蜜月精品| 亚洲午夜一区二区| 亚洲国产美国国产综合一区二区| 99国产精品| 亚洲黄页视频免费观看| 午夜日韩在线| 亚洲与欧洲av电影| 欧美激情欧美激情在线五月| 久久久久天天天天| 国产精品免费视频xxxx| 亚洲伦理自拍| 亚洲免费观看高清完整版在线观看熊 | 美女精品自拍一二三四| 欧美日本国产在线| 亚洲美女av在线播放| 日韩图片一区| 欧美激情在线免费观看| 亚洲国产精品ⅴa在线观看| 精品99一区二区| 久久久久久97三级| 欧美在线综合视频| 国产自产在线视频一区| 久久成人精品| 欧美电影在线免费观看网站 | 欧美大片专区| 亚洲日本免费电影| 亚洲欧美伊人| 国产日韩欧美一区二区三区四区| 亚洲一区二区三区免费观看| 久久国产精品99久久久久久老狼| 国产精品mm| 久久精品1区| 亚洲精品久久久久久一区二区| 一区二区三区**美女毛片| 亚洲欧洲一区二区三区久久| 欧美ed2k| 正在播放亚洲一区| 欧美成人精品h版在线观看| 亚洲精品国产系列| 国产日韩精品一区二区三区| 久久伊人精品天天| 亚洲一区二区三区免费视频| 欧美成年网站| 久久精品人人做人人爽| 亚洲一区亚洲| aa级大片欧美| 亚洲精品欧美在线| 在线欧美三区|