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

洛譯小筑

別來無恙,我的老友…
隨筆 - 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>
            国内伊人久久久久久网站视频| 亚洲级视频在线观看免费1级| 一区二区精品| 亚洲第一精品电影| 欧美极品欧美精品欧美视频| 91久久久久久国产精品| 亚洲国产综合在线看不卡| 欧美视频一区二区三区…| 亚洲一区二区精品在线| 亚洲综合欧美日韩| 亚洲成人资源网| 99国产精品久久| 国产一区91精品张津瑜| 欧美高清视频在线| 欧美日韩亚洲国产精品| 午夜伦理片一区| 久久精品国产一区二区电影| 亚洲黄色免费网站| 亚洲精品国产精品国自产观看浪潮 | 欧美日韩国产精品成人| 99re热精品| 日韩午夜av| 国产精品免费一区豆花| 久久午夜精品一区二区| 欧美激情精品| 欧美一区二区久久久| 久久男人资源视频| 亚洲伦理久久| 午夜欧美视频| 亚洲欧洲美洲综合色网| 亚洲午夜羞羞片| 在线看视频不卡| 中文精品视频一区二区在线观看| 国产精品亚洲第一区在线暖暖韩国| 久久亚洲精品伦理| 欧美日韩在线观看视频| 欧美在线观看视频在线| 欧美大片一区| 久久国产精品亚洲va麻豆| 久久男女视频| 亚洲欧美日韩成人| 美日韩在线观看| 亚洲在线视频观看| 亚洲精品中文在线| 国产伦精品一区二区三| 久久精品av麻豆的观看方式| 欧美福利视频在线观看| 欧美一区二区视频网站| 欧美精品一区二区三区一线天视频| 久久国产免费| 欧美日韩三级| 欧美韩日一区二区三区| 国产精品美女www爽爽爽视频| 欧美高清不卡在线| 国产亚洲一区二区三区| 国产精品99久久久久久有的能看| 男男成人高潮片免费网站| 欧美黑人国产人伦爽爽爽| 亚洲久久成人| 久久久久欧美精品| 欧美综合国产| 国产精品久久一区二区三区| 亚洲国产日韩欧美| 亚洲大胆人体视频| 午夜亚洲视频| 午夜精品久久久| 欧美亚洲成人精品| 欧美大片在线观看一区二区| 在线国产日韩| 久久最新视频| 久久综合久久综合这里只有精品| 欧美性生交xxxxx久久久| 亚洲精品免费网站| 一本色道久久综合| 欧美人与禽猛交乱配| 亚洲高清在线| 99成人免费视频| 欧美经典一区二区三区| 亚洲激精日韩激精欧美精品| 亚洲精品乱码久久久久久按摩观 | 欧美专区日韩专区| 国产精品亚洲一区| 欧美一区二区黄色| 久久午夜国产精品| 在线观看亚洲| 欧美激情二区三区| 亚洲精品资源| 午夜在线电影亚洲一区| 国产精品久久影院| 久久高清福利视频| 欧美大尺度在线| 夜夜爽av福利精品导航 | 亚洲午夜性刺激影院| 亚洲免费视频网站| 国产亚洲亚洲| 久久综合伊人| 欧美国产精品v| 亚洲日本成人在线观看| 欧美精品免费在线| 亚洲私人影院| 久久久综合免费视频| 91久久香蕉国产日韩欧美9色| 欧美不卡高清| 亚洲自拍偷拍网址| 欧美高清视频一区二区三区在线观看 | 午夜精品在线看| 欧美成人午夜影院| 一区二区三区偷拍| 国内精品久久久久久久影视蜜臀 | 久久精品日韩欧美| 亚洲欧洲在线看| 久久精品日韩一区二区三区| 在线看片成人| 国产精品美女久久久久久2018| 欧美在线视频在线播放完整版免费观看 | 久久精品中文| 一本大道久久精品懂色aⅴ| 黄色亚洲网站| 欧美日韩国产a| 久久香蕉国产线看观看av| 亚洲国产导航| 欧美亚洲一区| 在线成人激情视频| 国产一区二区在线观看免费| 欧美成人国产一区二区| 亚洲综合色婷婷| 亚洲国产天堂久久综合| 久久人人爽爽爽人久久久| 亚洲视频999| 91久久精品国产91久久性色| 国产麻豆精品视频| 欧美日韩亚洲一区二区三区在线| 久久视频在线免费观看| 亚洲性视频网址| 亚洲人成在线观看| 欧美成人日本| 蜜臀99久久精品久久久久久软件 | 欧美另类变人与禽xxxxx| 亚洲男人的天堂在线| 亚洲福利视频网站| 久久久久综合网| 欧美在线精品免播放器视频| 亚洲一区二区三区在线| 99香蕉国产精品偷在线观看| 亚洲国产精品热久久| 在线视频国内自拍亚洲视频| 黑人一区二区| 国产亚洲精品成人av久久ww| 国产精品久久久久久久久久ktv | 一区二区三区日韩在线观看 | 亚洲欧美国产高清va在线播| 亚洲性xxxx| 亚洲视频香蕉人妖| 一区二区三区国产精华| 99国产欧美久久久精品| 日韩视频一区二区三区在线播放| 亚洲福利av| 亚洲国产一区二区a毛片| 亚洲黄色一区| 日韩一级黄色大片| 日韩午夜精品视频| 亚洲深夜福利网站| 亚洲欧美另类国产| 午夜精品久久久久| 亚洲一区在线免费| 欧美一区午夜精品| 久久久99免费视频| 美日韩精品视频免费看| 欧美制服丝袜第一页| 亚洲在线一区二区| 在线亚洲免费| 亚洲午夜精品福利| 亚洲欧美国产日韩中文字幕| 亚洲视频www| 老司机一区二区三区| 欧美在线观看网站| 你懂的视频欧美| 亚洲国产三级在线| 一区二区三区福利| 小黄鸭精品aⅴ导航网站入口| 欧美一级视频精品观看| 久久综合伊人77777| 欧美日韩国产综合久久| 国产视频欧美视频| 亚洲电影免费| 亚洲一区日韩在线| 久久人人97超碰精品888| 欧美激情免费观看| 欧美精品一级| 欧美日韩精品伦理作品在线免费观看 | 亚洲一二三区精品| 久久久午夜电影| 日韩视频精品在线观看| 亚洲欧洲一区| 久久精品国产视频| 欧美激情国产高清| 国产精品扒开腿做爽爽爽软件| 欧美美女bb生活片| 国产日韩在线看| 91久久在线播放|