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

洛譯小筑

別來無恙,我的老友…
隨筆 - 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>
            欧美日韩在线播放一区二区| 亚洲专区欧美专区| 亚洲激情在线激情| 精品成人在线观看| 伊人久久亚洲美女图片| 1769国内精品视频在线播放| 伊人久久综合97精品| 黄色在线一区| 亚洲精品系列| 亚洲免费视频网站| 午夜国产一区| 狂野欧美激情性xxxx| 亚洲国产91精品在线观看| 久久人体大胆视频| 欧美激情视频一区二区三区在线播放 | 美国十次成人| 欧美激情一区二区| 国产精品第一区| 国产精品日韩专区| 亚洲福利一区| 午夜免费在线观看精品视频| 六月婷婷一区| 亚洲视频在线观看| 免费成人av资源网| 国产欧美精品在线观看| 亚洲精品在线一区二区| 欧美夜福利tv在线| 亚洲日本成人网| 久久精品国产久精国产爱| 欧美日韩国产成人高清视频| 国内外成人免费激情在线视频| 亚洲精品综合在线| 久久婷婷麻豆| 亚洲欧美日本在线| 欧美日韩一本到| 亚洲国产日韩欧美| 久久久欧美一区二区| 亚洲无玛一区| 欧美日韩精品免费观看视频| 在线播放日韩欧美| 久久国产一区二区| 亚洲激情欧美激情| 亚洲淫性视频| 欧美高清在线视频| 欧美在线观看你懂的| 国产精品夫妻自拍| 一二三四社区欧美黄| 欧美成人一区二区三区| 欧美综合二区| 国产婷婷色一区二区三区四区| 亚洲视频导航| 日韩视频在线永久播放| 欧美激情在线| 日韩一级精品| 亚洲精品欧美| 欧美日韩国产综合在线| 99精品国产高清一区二区| 欧美激情一区二区三区四区| 久久久亚洲午夜电影| 激情国产一区| 免费日韩av| 蜜桃久久av| 亚洲三级电影全部在线观看高清| 欧美成人免费观看| 欧美成人激情视频| 日韩一级片网址| 日韩视频精品在线观看| 欧美日韩在线一区二区三区| 亚洲一二三区在线观看| 亚洲无线观看| 国产在线拍揄自揄视频不卡99 | 国产一区二区你懂的| 久久久久久久久久久一区| 欧美一区二区三区在线| 韩国欧美一区| 亚洲福利专区| 欧美日本精品在线| 亚洲欧美日韩系列| 久久精品国产一区二区电影| 91久久国产综合久久| 亚洲精品小视频在线观看| 国产精品ⅴa在线观看h| 久久精品国产99精品国产亚洲性色| 久久成人免费电影| 最新日韩在线视频| 宅男噜噜噜66国产日韩在线观看| 国产精品久久久久久av下载红粉| 新狼窝色av性久久久久久| 久久国产天堂福利天堂| 亚洲精选视频免费看| 这里只有精品丝袜| 在线看片一区| 亚洲无亚洲人成网站77777| 狠狠色狠狠色综合人人| 亚洲精品一区二区三区福利| 国产日韩欧美一区在线| 亚洲福利专区| 国产一区二区三区四区三区四 | 亚洲愉拍自拍另类高清精品| 欧美一区二区久久久| 欧美日本国产在线| 亚洲欧美日韩在线一区| 久久看片网站| 欧美一激情一区二区三区| 欧美高清成人| 久久精品国产一区二区电影| 欧美精品免费播放| 久久综合伊人77777麻豆| 欧美日韩国产系列| 狂野欧美一区| 国产精品五月天| 亚洲精品中文字幕在线| 一区在线观看| 性欧美xxxx视频在线观看| 一区二区不卡在线视频 午夜欧美不卡在 | 久久国产精品网站| 亚洲一区二区精品在线观看| 久久久亚洲欧洲日产国码αv| 亚洲欧美日韩中文在线制服| 欧美国产极速在线| 免费亚洲一区二区| 国内精品一区二区三区| 亚洲欧美在线一区二区| 亚洲字幕一区二区| 欧美午夜片在线观看| 亚洲伦理精品| 99精品国产一区二区青青牛奶| 久久亚洲国产精品日日av夜夜| 欧美专区在线观看| 国产精品美女久久久| 在线亚洲精品| 性8sex亚洲区入口| 国产日韩亚洲欧美精品| 亚洲欧美日韩人成在线播放| 午夜精品久久久久久久久久久| 欧美午夜精品伦理| 亚洲免费综合| 久久国产乱子精品免费女| 国产精品五月天| 亚洲欧美一区二区原创| 久久av最新网址| 国产一区二区精品久久91| 欧美一级淫片aaaaaaa视频| 欧美一区二区精品久久911| 国产精品久久久久久久久久直播| 一片黄亚洲嫩模| 午夜精品久久久久久久| 国产农村妇女精品一二区| 欧美专区在线| 欧美国产精品va在线观看| 亚洲精品久久久久久一区二区| 欧美精品一区二区久久婷婷| 亚洲美女黄色| 香蕉久久夜色| 激情久久婷婷| 欧美成人嫩草网站| 日韩一级大片| 久久国产一区| 亚洲精品在线三区| 国产精品人人爽人人做我的可爱| 亚洲婷婷在线| 久久伊人一区二区| 亚洲毛片在线看| 国产精品亚洲综合久久| 久久乐国产精品| 一区二区三区日韩精品视频| 久久精品亚洲一区| 美女主播一区| 中文国产成人精品| 久久久久久久久久久一区| 亚洲人屁股眼子交8| 国产精品久久久久久亚洲毛片 | 另类天堂av| 一区二区高清视频在线观看| 国产欧美精品在线| 欧美高清在线视频观看不卡| 亚洲综合国产精品| 亚洲第一主播视频| 性欧美暴力猛交另类hd| 亚洲人成欧美中文字幕| 国产精品永久入口久久久| 免费成人性网站| 亚洲欧美综合另类中字| 最新亚洲电影| 久久五月天婷婷| 亚洲一区二区三区高清| 亚洲国产精品高清久久久| 国产精品综合不卡av| 你懂的网址国产 欧美| 亚洲视频一起| 亚洲精品乱码久久久久久久久| 久久米奇亚洲| 亚洲一区二区三区四区五区午夜| 亚洲国产精品ⅴa在线观看| 国产日韩在线不卡| 欧美四级在线观看| 欧美激情一二区| 欧美国产日韩一区二区| 久久精品视频va| 欧美一区二区三区免费视|