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

洛譯小筑

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

[ECPP讀書筆記 條目14] 要注意資源管理類中的復制行為

條目13中介紹了“資源獲取即初始化”(Resource Acquisition Is Initialization,簡稱RAII)的概念,它是資源管理的中心內容。同時條目13中還使用auto_ptrtr1::shared_ptr作為示例,描述了如何利用這一概念來管理堆上的資源。然而并不是所有的資源都分配于堆上,對于不分配于堆上的資源,諸如auto_ptrtr1::shared_ptr這一類的智能指針并不適合于處理它們。這是千真萬確的,你必須不時地自己動手,創建自己的資源管理類。

舉例說,你正使用一個C版本的API所提供的lockunlock函數來處理Mutex類型的互斥對象:

void lock(Mutex *pm);              // 通過pm為互斥量上鎖

void unlock(Mutex *pm);            // 為互斥量解鎖

為了確保你曾上鎖的互斥量都得到解鎖,你應該自己編寫一個類來管理互斥鎖。這樣的類的基本結構應遵循RAII的原理,那就是:資源在構造過程中獲得,在析構過程中釋放:

class Lock {

public:

  explicit Lock(Mutex *pm)

  : mutexPtr(pm)

  { lock(mutexPtr); }                   // 獲取資源

 

  ~Lock() { unlock(mutexPtr); }         // 釋放資源

 

private:

  Mutex *mutexPtr;

};

客戶通過傳統的RAII風格來使用Lock類:

Mutex m;                           // 定義互斥量以便使用

...

{                                  // 創建程序塊用來定義臨界區

 Lock ml(&m);                      // 為互斥量上鎖

...                                // 進行臨界區操作

}                                  // 在程序塊末尾互斥量將自動解鎖

這樣可以正常工作,但是如果復制一個Lock對象,將會發生些什么呢?

Lock ml1(&m);                      // m上鎖

 

Lock ml2(ml1);                     // ml1復制給ml2

                                   // 將會發生什么呢?

有一個問題是所有的RAII類創建者必須面對的,那就是:當復制一個RAII對象時應做些什么。以上是對于這個一般化問題的一個較具體的示例。大多數時候,以下四種可行的方案供你選擇。

禁止復制。在許多情況下,允許RAII被復制沒有任何意義。比如對于Lock類來說就是這樣,因為復制同步原型在大多數情況下都沒有什么意義。當復制一個RAII類無意義時,你就應該禁止它。條目6中詳細介紹了實現方法:將拷貝賦值運算符聲明為私有的。對于Lock而言,應該是下面的情形:

class Lock: private Uncopyable {      // 防止復制 參見條目6

public:

 ...                                   // 同上

};

為潛在生成的資源進行引用計數。有時,我們期望能保留對一個資源的所有權,直到其所涉及的最后一個對象被刪除為止。在這種情況下,復制一個RAII對象將會添加一個引用資源對象的計數。這就是tr1::shared_ptr所使用的“復制”的含義。

通常情況下,RAII類可以通過包含一個tr1::shared_ptr數據成員來實現引用計數復制行為。舉例說,如果Lock在設計時之初就期望使用引用計數,它可能會用tr1::shared_ptr<Mutex>代替Mutex*來作為mutexPtr的類型。但是不幸的是,tr1::shared_ptr默認的行為是:當引用計數值變為零時,刪除其所指向的內容,但這不是我們想要的。當一個Mutex用完時,我們希望對其進行的操作是解鎖,而不是刪除它。

所幸的是,tr1::shared_ptr允許指定一個“刪除器”,它是一個函數或一個函數對象,用于在引用計數值為零時進行調用。(auto_ptr并不包含這一特性,它總是刪除它所指向的內容。)刪除器是tr1::shared_ptr構造函數的第二個(可選的)參數,所以代碼應該是這樣的:

class Lock {

public:

  explicit Lock(Mutex *pm)     // 初始化shared_ptr,參數為

  : mutexPtr(pm, unlock)       // 指向Mutex的指針和解鎖函數

 

    lock(mutexPtr.get());      // 關于"get"的信息請參見條目15

  }

 

private:

  std::tr1::shared_ptr<Mutex> mutexPtr;

};                             // 使用shared_ptr而不是原始指針

在本示例中,請注意Lock類不再聲明析構函數。這是因為我們不再需要它了。條目5中介紹了類的析構函數(無論是編譯器自動生成的還是用戶自定義的)會自動為類的非靜態數據成員進行析構。就像本示例中的mutexPtr。然而,當互斥量的引用計數變為零時,mutexPtr將會自動調用tr1::shared_ptr的刪除器unlock。(此時如果你為代碼添加了一段注釋,告訴人們你并沒有忘記編寫析構函數,你只是借助了默認的編譯器行為。人們看了這樣的注釋思路會更清晰一些。他們會感激你的。)

復制潛在生成的資源。一些時候,你可以在需要的情況下為資源復制出任意份數的副本,此時你需要一個資源管理類的唯一理由就是:確保每份副本在其工作完成之后得到釋放。在這種情況下,復制資源管理對象的同時,也要復制出其所涉及的資源。也可以說,復制一個資源管理對象時,將進行“深度復制”。

標準string類型的一些實現版本中,包含著一個指向堆內存的指針,這個指針所指向的就是字符串所保存的位置。這樣的string對象包含著一個指向堆內存的指針。當一個string對象被復制時,將同時復制這一指針和其指向的內存。這樣的string就進行了一次深度復制。

傳遞潛在生成資源的所有權。在少數情況下,你可能需要確保僅僅有一個RAII對象引用了一個原始的資源,當復制這一RAII對象時,資源的所有權也將從源對象傳遞到目標對象。如同條目13中所解釋的,這是通過auto_ptr所實現的“復制”的含義。

拷貝函數(拷貝構造函數和拷貝賦值運算符)可以由編譯器自動生成,但是如果編譯器自動生成版本無法滿足你的需要(條目5中解釋了C++的默認行為),你就應該自己編寫這些函數。在一些情況下,你可能還會需要支持這些函數的一般化的版本。這些版本將在條目45中介紹。

時刻牢記

由于復制一個RAII對象必須要同時復制其所管理的資源,因此資源的復制行為決定RAII對象的復制行為。

RAII類有兩種一般性的復制行為:禁止復制和進行資源計數。同時其他的行為也是可能存在的。

posted on 2007-05-11 18:40 ★ROY★ 閱讀(967) 評論(1)  編輯 收藏 引用 所屬分類: Effective C++

評論

# re: 【翻譯】[Effective C++第三版?中文版][第14條]要留心資源管理類中的復制行為  回復  更多評論   

boost的scoped_ptr使用的第一個策略:禁止復制

:)
2007-05-23 10:15 | recorder
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美a级理论片| 欧美精品综合| 美国成人毛片| 午夜精品在线看| 亚洲精品色图| 一区二区三区精品久久久| 999亚洲国产精| 一本色道久久综合亚洲二区三区| 精品9999| 亚洲精品在线一区二区| 一区二区三区高清视频在线观看| 亚洲人成艺术| 午夜精彩国产免费不卡不顿大片| 香蕉久久国产| 欧美11—12娇小xxxx| 99riav久久精品riav| 亚洲少妇中出一区| 美女任你摸久久| 国产精品嫩草影院av蜜臀| 国产精品一区二区久久| 亚洲欧洲一区二区在线观看| 亚洲一区在线观看视频 | 亚洲免费一在线| 久久久久成人精品免费播放动漫| 欧美激情在线免费观看| 午夜在线一区二区| 欧美日韩国产在线| 一本色道久久88综合亚洲精品ⅰ| 欧美日韩国产成人精品| 狠狠综合久久av一区二区小说| 亚洲精品一区二区三区福利 | 国产精品久久久久久久午夜 | 欧美在线资源| 亚洲一区二区毛片| 欧美性猛片xxxx免费看久爱| 在线亚洲伦理| 亚洲午夜激情免费视频| 国产精品夜夜夜一区二区三区尤| 亚洲手机在线| 亚洲欧美日韩中文视频| 国产精品一区二区久久久久| 久久久噜噜噜久久| 老鸭窝毛片一区二区三区| 尤物视频一区二区| 亚洲肉体裸体xxxx137| 久久亚洲一区二区| 日韩视频在线观看一区二区| 亚洲三级国产| 国产原创一区二区| 亚洲福利视频免费观看| 欧美性事免费在线观看| 久久久成人网| 欧美区亚洲区| 欧美一级在线视频| 蜜臀va亚洲va欧美va天堂| 一本大道久久a久久综合婷婷| 一区二区三区日韩| 伊人夜夜躁av伊人久久| 亚洲神马久久| 中文精品视频| 久久福利视频导航| 久久久综合网站| 午夜在线精品| 欧美精品v国产精品v日韩精品 | 国产精品久久久999| 欧美第一黄色网| 国内精品久久久久久 | 一区二区三区不卡视频在线观看| 亚洲线精品一区二区三区八戒| 在线精品视频一区二区三四| 亚洲在线观看视频| 亚洲男人第一网站| 国产精品国产三级国产专区53| 亚洲欧洲另类国产综合| 亚洲人www| 欧美精品一区二区三区四区| 欧美国产视频日韩| 日韩视频―中文字幕| 亚洲一区二区在线观看视频| 欧美高清不卡在线| 亚洲精选成人| 午夜欧美视频| 国内精品久久久久影院薰衣草| 午夜精品久久久久久久99水蜜桃 | 国产精品magnet| 亚洲欧美国产日韩中文字幕| 午夜精品一区二区三区在线播放 | 欧美国产第一页| 中文日韩在线视频| 久久久久久夜精品精品免费| 在线日韩中文| 国产精品视频男人的天堂| 欧美在线观看一区二区| 欧美视频官网| 久久se精品一区精品二区| 嫩草影视亚洲| 欧美亚洲一区在线| 一区二区精品在线| 好吊色欧美一区二区三区四区| 欧美激情亚洲视频| 久久国产精品色婷婷| 亚洲天堂成人| 在线一区二区视频| 亚洲精品视频在线播放| 老司机免费视频一区二区三区| 亚洲欧美日本视频在线观看| 亚洲人成亚洲人成在线观看| 好吊色欧美一区二区三区四区| 欧美日韩视频| 欧美三日本三级三级在线播放| 男女激情视频一区| 欧美精彩视频一区二区三区| 久热国产精品| 欧美国产精品中文字幕| 麻豆国产精品va在线观看不卡| 久久精品国产亚洲一区二区三区| 亚洲一区二区在线视频| 亚洲午夜精品视频| 欧美一区免费| 欧美国产丝袜视频| 国产精品久久久久久久久免费樱桃| 欧美午夜在线观看| 国产视频在线观看一区二区三区| 国产伦精品一区二区三区四区免费 | 国产精品久久久久久久久久三级| 欧美精品午夜| 国产欧美精品xxxx另类| 好看的av在线不卡观看| 亚洲国产精品999| 一区二区三区视频观看| 国产精品久久久久av免费| 国产精品综合网站| 亚洲激情视频在线| 欧美成人午夜免费视在线看片| 亚洲一级片在线观看| 免费欧美日韩国产三级电影| 欧美人交a欧美精品| 精品96久久久久久中文字幕无| 99人久久精品视频最新地址| 久久久国产亚洲精品| 国产精品自在欧美一区| 亚洲欧美日韩在线一区| 亚洲久久成人| 欧美国产日韩精品| 亚洲激情在线激情| 欧美激情自拍| 久久综合色播五月| 国产午夜精品美女视频明星a级 | 亚洲在线观看视频网站| 欧美福利在线| 欧美日韩国产限制| 亚洲国产精品一区二区久| 美女黄网久久| 9色精品在线| 亚洲理论在线| 国内自拍视频一区二区三区| 老司机aⅴ在线精品导航| 久久人人97超碰人人澡爱香蕉| 美女国产精品| 一区二区在线观看视频| 中文在线资源观看网站视频免费不卡 | 麻豆成人av| 午夜精品久久久久久久久久久久久 | 欧美在线免费观看| 羞羞色国产精品| 亚洲国产欧美一区| 亚洲第一免费播放区| 免费在线播放第一区高清av| 亚洲电影免费在线观看| 亚洲美女淫视频| 在线观看日韩av先锋影音电影院| 日韩午夜精品| 亚洲精品一二三| 久久在线91| 亚洲肉体裸体xxxx137| 夜夜爽www精品| 国产精品swag| 久久综合给合久久狠狠色| 国产精品福利网| 亚洲国产精品久久人人爱蜜臀| 国产精品推荐精品| 一二三区精品| 性欧美18~19sex高清播放| 欧美黑人多人双交| 亚洲第一精品福利| 亚洲国产日韩一级| 乱码第一页成人| 女人香蕉久久**毛片精品| 国产一区激情| 久久蜜桃资源一区二区老牛| 在线一区二区三区做爰视频网站 | 欧美在线视频日韩| 午夜日韩在线观看| 国产精品成人一区二区三区吃奶| 亚洲国产欧美在线人成| 亚洲二区在线| 欧美美女操人视频| 亚洲午夜一区| 久久久国产精品亚洲一区| 亚洲国产乱码最新视频|