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

Shuffy

不斷的學(xué)習(xí),不斷的思考,才能不斷的進(jìn)步.Let's do better together!
posts - 102, comments - 43, trackbacks - 0, articles - 19

[轉(zhuǎn)]http://m.shnenglu.com/tiandejian/archive/2007/04/22/ecpp_08.html
第8條:
防止因異常而中止析構(gòu)函數(shù)

C++ 沒(méi)有禁止析構(gòu)函數(shù)引發(fā)異常,但 C++ 分不推薦這一做法。這樣做有充足的理由。請(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)生無(wú)法預(yù)知的行為。本示例將產(chǎn)生無(wú)法預(yù)知的行為。在使用其它的標(biāo)準(zhǔn)庫(kù)容器(比 list set 等),任意的 TR1 器(參見(jiàn) 54 ),甚至是一個(gè)數(shù)組,同樣都會(huì)產(chǎn)生無(wú)法預(yù)知的行為。然而為你帶來(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ù)的某一操作可能失敗,并且有可能對(duì)這一失敗的操作拋出一個(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;

};

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

{                              // 開(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ú)法運(yùn)行下去了,上面的方法就是一個(gè)可行的選擇。如果允許析構(gòu)函數(shù)傳播異常將導(dǎo)致程序行為無(wú)法預(yù)知,這樣做的優(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)處理由 DBConnection close 產(chǎn)生的異常。這樣做還可以保持跟蹤 DBConnection 所建立的連接是否被 DBConnection 自己的 close 函數(shù)正常關(guān)閉,如果關(guān)閉失敗則在 DBConn 的析構(gòu)函數(shù)再次嘗試。這可以防止已建立的連接發(fā)生泄漏。然而,如果在 DBConn 的析構(gòu)函數(shù) [1] 中對(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)閉連接,

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

   }

   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)單易用”這一忠告(見(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ú)法避免不成熟的終止和未知行為。在上邊的示例中,讓客戶端程序員自己手動(dòng)調(diào)用 close 并不會(huì)為其帶來(lái)過(guò)多的負(fù)擔(dān),相反地,這樣做為客戶端程序員提供了處理錯(cuò)誤的機(jī)會(huì),否則他們就會(huì)束手無(wú)策。如果他們沒(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ò)誤,是他們自己選擇不去利用它的。

需要記住的

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

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



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

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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精品国产| 欧美高清不卡在线| 亚洲国产精品v| 一区二区欧美亚洲| 欧美一区国产一区| 久久久久综合| 国产色综合久久| av成人免费在线观看| 久久久亚洲午夜电影| 欧美伊人久久久久久午夜久久久久| 欧美国产视频一区二区| 亚洲日本免费电影| 亚洲欧美一区二区三区在线 | 欧美精品激情| 国产一区二区按摩在线观看| 亚洲精品国产欧美| 久久国产福利国产秒拍| 最新成人av网站| 欧美一级精品大片| 国产精品视频xxxx| 亚洲午夜一区| 一区二区冒白浆视频| 欧美va天堂| 日韩视频免费在线| 亚洲国产精品成人| 欧美成人蜜桃| 亚洲一区二区三区精品在线观看| 最新亚洲电影| 国产精品一区一区三区| 亚洲欧美自拍偷拍| 一区二区三区你懂的| 国产女主播一区二区| 久久久久久亚洲综合影院红桃 | 狠狠爱综合网| 久久久久国产精品一区二区| 欧美一区成人| 亚洲日本va午夜在线影院| 亚洲国产专区校园欧美| 欧美高清视频| 欧美日韩另类在线| 久久精品国产v日韩v亚洲| 亚洲欧美99| 最新日韩在线视频| 亚洲欧美日韩在线高清直播| 韩国久久久久| 99视频有精品| 伊人久久久大香线蕉综合直播 | 亚洲精品视频免费在线观看| 亚洲精品国产精品国自产在线| 国产精品白丝黑袜喷水久久久| 蜜桃精品久久久久久久免费影院| 欧美韩日亚洲| 老司机午夜精品视频| 国产日韩精品一区| 99精品福利视频| av成人动漫| 欧美午夜不卡在线观看免费| 裸体歌舞表演一区二区| 国产综合网站| 欧美亚洲三级| 久久久亚洲高清| 激情久久中文字幕| 久久精品综合一区| 久久精品人人做人人爽| 国产一区二区欧美日韩| 欧美一区二区三区在线播放| 欧美一区激情| 红桃视频国产精品| 欧美日韩精品在线| 亚洲欧美国产高清va在线播| 欧美一区二区三区久久精品茉莉花 | 欧美承认网站| 亚洲国产精品一区二区www| 欧美专区在线播放| 亚洲福利免费| 亚洲国产电影| 欧美超级免费视 在线| 欧美福利小视频| 亚洲免费在线观看视频| 国产在线精品成人一区二区三区| 欧美一区二区三区日韩视频| 亚洲一区免费| 亚洲国产精品毛片| 欧美日韩在线精品一区二区三区| 亚洲国产欧洲综合997久久| 99riav国产精品| 国产精品日日摸夜夜添夜夜av| 亚洲欧美在线高清| 玖玖在线精品| 亚洲一区二区三区在线| 国产亚洲欧美aaaa| 欧美美女日韩| 香蕉国产精品偷在线观看不卡 | 欧美日韩高清在线一区| 亚洲一级黄色片| 欧美3dxxxxhd| 亚洲欧美日韩视频二区| 亚洲福利久久| 国产精品影音先锋| 欧美成人久久| 久久久蜜臀国产一区二区| 美国十次成人| 亚洲在线播放电影| 在线观看日韩精品| 黑人极品videos精品欧美裸| 国产精品欧美久久久久无广告| 欧美好吊妞视频| 欧美日本国产在线| 欧美高清在线播放| 国产日本亚洲高清| 久热精品视频在线观看一区| 欧美在线|欧美| 亚洲一区二区三区视频播放| 亚洲天堂网在线观看| 亚洲欧美在线磁力| 久久久噜噜噜久久久| 欧美二区在线| 欧美私人啪啪vps| 国产一区香蕉久久| 亚洲欧洲一区二区三区久久| 国产一区二区三区的电影 | 国语自产精品视频在线看8查询8 | 日韩一区二区电影网| 亚洲在线视频观看| 久久久久久香蕉网| 久热精品视频在线观看| 欧美精品成人在线| 国产麻豆精品视频| 亚洲成人在线免费| 亚洲国产美女| 亚洲免费在线播放| 玖玖综合伊人| 99精品视频一区| 美女视频黄 久久| 国产精品久久网| 亚洲高清在线观看| 久久久久久**毛片大全| 夜夜精品视频| 国产精品福利影院| 亚洲美女中文字幕| 亚洲国产精品成人va在线观看| 久久国产精品黑丝| 韩国一区二区在线观看| 欧美亚洲免费电影| 亚洲欧美日韩国产| 校园激情久久| 国产精品免费小视频| 亚洲精品国精品久久99热一| 国产欧美日韩中文字幕在线| 欧美一区二区黄| 久久激情综合网| 在线观看成人av电影| 亚洲黄色精品| 国产精品美腿一区在线看| 亚洲午夜精品久久久久久浪潮| 亚洲一区视频在线| 国产综合久久久久久| 亚洲国产美女精品久久久久∴| 欧美日韩成人免费| 久久精品五月| 欧美日本在线看| 久久精品国产77777蜜臀 | 免费的成人av| 欧美日韩在线免费观看| 久久午夜电影| 国产精品羞羞答答xxdd| 欧美高清hd18日本| 国产欧美日韩另类视频免费观看| 久久嫩草精品久久久精品| 欧美日韩一区二区在线观看视频| 久久国产精品久久国产精品| 欧美日韩一级大片网址| 欧美成人一品| 亚洲国产二区| 欧美成人一区在线| 欧美激情无毛| 亚洲精品日韩综合观看成人91| 久久久久久久久综合| 欧美mv日韩mv国产网站| 黄色日韩在线| 欧美gay视频| 亚洲精品国产视频| 亚洲一区二区三区欧美 | 一区二区三区高清在线| 亚洲欧洲日本mm| 欧美日韩www| 亚洲小视频在线观看| 欧美一区日韩一区| 狠狠色伊人亚洲综合成人| 欧美一级视频免费在线观看| 欧美午夜一区二区三区免费大片|