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

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

C++引用計數的智能指針有效回收方法

  怎樣從災難性故障中,恢復一個長期運行、系統級的后臺守護進程或者服務,在如今的軟件設計過程中,已成為了一個重要的考慮因素。當這些軟件是由C++語言編成,并使用了引用計數的智能指針時,那么,智能指針的有效回收,對系統是否具有可伸縮級的恢復能力、甚至正確地繼續未完成的操作來說,都顯得至關重要。

  在本文中,描述了一種方法,可從關機之后的軟件恢復中,有效地回收引用計數指針,而且此方法在內存占用方面也非常高效,這種方法的關鍵在于避免對象復制,而對象復制通常是由C++中指針引用的串行化與反串行化這種傳統技術產生的。當從存檔文件中反串行化時,本方法使用了標記(tag)來唯一地識別指針對象,且在系統恢復時由一個對象緩存來保存指針引用。

  本文以一個基于事件的商業實時作業調度系統來進行演示,其通常由大型市場咨詢公司使用,每天都會在集群工作站上處理數不勝數的計算任務。

  為什么許多C++軟件項目會使用自動內存管理技術呢,因為它有以下好處:

  2 代碼安全性。避免了太早釋放一個對象所帶來的風險。

  2 代碼正確性。避免了忘記釋放未使用內存所帶來的風險。

  2 代碼模塊性。代碼中不再需要點綴著與程序無關的簿記代碼。

  2 編程簡單性?,F在可假定一種無限內存的計算模式。

  2 編程高效性。程序員不再擔心內存管理問題。

  引用計數智能指針,有時也稱為計數體術語,是一種生命期受管的對象,其對引用它的數量,有一個內部的計數器。當內部引用計數為零時,這些對象會自動銷毀自身,這是一種非常有用的技術,已運用在許多C++軟件產品項目中,因為簡單易行,且無需對語言或編譯器進行任何擴展。

  引用計數智能指針能進一步定義為一體式或分離式,一體式智能指針把引用計數放在自身內,而分離式智能指針則把引用計數放在對象之外。在本文中,使用的是分離式智能指針方案,這需要在訪問實際對象指針之前,在智能指針模板對象中重載 -> * 操作符,從本質上來說,這也是代理(Proxy)設計模式的一個特例。

  就目前來說,還沒有一種方案以高效利用內存的方式描述了怎樣恢復智能指針,而傳統的C++對象串行與反串行化方法,會導致內存低效,因為當一個反串行化的對象遇到一個對它的引用時,總是會創建一個新對象,在最壞的情況下,這會把一個恢復后的守護進程的內存消耗量,推到一個無法接受的高度,致使它無法繼續運行下去。

  問題的引出

  傳統對象的串行與反串行化方案,也能實現智能指針,只不過在內存上比較低效而已。在這些傳統方案中,當一個對象串行化時,對象內的成員指針被解引用,它的內容與對象一起串行進存檔文件中。這種方法的問題在于,當反串行化時,成員指針會再次構造,且是每個恢復的對象都會這樣。

  下面以基于事件的作業調度系統來進行講解,作業定義在CJobDef對象中,其包含了作業的靜態屬性,如它執行的命令、工作目錄、及作業執行時的用戶ID。而作業定義的運行實例則包裝在CJobInst對象中,其包含了一些與實例有關的屬性,如它的進程ID、執行參數、及運行歷史記錄。在類層次上,每個CJobInst對象都包含了一個成員,其引用到觸發這次作業實例的原始CJobDef對象。

  圖1是軟件停止運行之前的系統,運行時CJobInst對象的多個實例可能會引用至同一個CJobDef對象。在軟件停止及恢復后,傳統串行化對象恢復方法,會導致為每個運行的CJobInst對象,都創建一個CJobDef對象,如圖2中所示。

  clip_image001

  圖1:恢復之前的對象圖

  clip_image002

  圖2:內存低效恢復之后的對象圖

  這種情況發生在傳統的C++類對象中指針成員串行化與反串行化時,例1,是一段帶有重載>><<操作符,串行及反串行化CJobInstCJobDef類指針的CArchive類代碼,也證明了這點。

  例1

以下是引用片段:
classCJobDef
  {
  friendCArchive&operator>>(CArchive&ar,CJobDef*def)
  {
  ar>>def->command;
  }
  friendCArchive&operator<<(CArchive&ar,CjobDef*def)
  {
  ar<<def->command;
  }
  private:
  std::stringcommand;
  };
  classCJobInst
  {
  friendCArchive&operator>>(CArchive&ar,CJobInst*inst)
  {
  inst->m_def=newCJobDef;
  ar>>inst->m_def;
  }
  friendCArchive&operator<<(CArchive&ar,constCJobInst*inst)
  {
  ar<<inst->m_def;
  }
  private:
  CJobDef*m_def;
  };


  在CJobInst中串行化CJobDef的私有成員m_def涉及到調用CArchive類中適當的<<操作符,重載的<<操作符通過把對象屬性串行化進一個永久的存檔文件,來實現對CJobDef指針的串行化;反串行化CJobDef指針涉及到構造一個新的對象,并調用>>操作符從存檔文件中更新屬性。
 
  解決方案
 
  引用計數智能指針是由繼承自CReferable類一個對象實現的,其包含了一個私有引用計數器及用于修改其值的increaseReferenceCount()與decreaseReferenceCount()方法,而相應的Ref模板類,通過->*、= 操作符重載,也實現了訪問此對象及對生命期的管理。Ref模板對智能指針的賦值操作,會遞增對象的引用計數,而它的析構函數會遞減計數。智能指針中的對象只當它的引用計數為零時被銷毀。在上面的作業調度系統中,CJobDef對象被包裝在一個CJobDefPtr類型中,其由以下語句定義:
 

以下是引用片段:
typedefRef<CJobDef>CJobDefPtr;


這個CJobDefPtr類型,正是類CScheduler所用到的類型。當用戶提交一個作業到事件作業調度器時,會產生一個CJobDefPtr類型新的對象,且會賦予它CJobDef對象;此后,當作業實例創建時,也正是這個CJobDefPtr類型賦予給了實例。圖3演示了類CScheduler使用的CJobDefPtr類型。

  clip_image003

  圖3:作業定義類關系圖

  在CJobDefPtr類中,賦值=操作符遞增了CJobDef對象CReferable基類中的引用計數,而delete操作符遞減了這個引用計數。包裝在CJobDefPtr對象中的CJobDef對象不會被銷毀,直到它的引用計數為零,這也說明了在系統中,沒有其他任何對象引用CJobDef對象,它可以安全地被銷毀了。

  再次提醒,從作業中創建的作業實例,被包裝在一個CJobInst類中。與CJobDef一樣,類CScheduler只知道它對應版本的智能指針CJobInstPtr,而此對象的實例也會一直保持到沒有對它的引用為止。

  另外,在系統中,還包括了另外三個特性,以便使調度系統可高效地恢復:

  2 CReferable增加了一個tag屬性,以唯一地識別每個創建的指針實例,同時有一個getTag()方法可用于訪問此屬性。

  2 Ref模板類在稱為CReferableCache的全局對象緩存中管理它的對象,此全局對象緩存可由其他智能指針對象訪問。

  2 Ref模板類添加了一個impersonate()方法,其允許一個智能指針以給定的tag轉換為另一個智能指針。

  當一個新的CJobDefPtrCJobInstPtr被創建時,在CReferable基類構造函數中,會分配給對象唯一的一個tag。這個tag可由幾種方式產生,但任一種方式都必須保證在每次軟件運行時,都會有一個唯一的ID。一個簡單的方案是使用一個靜態、全局的計數器對象,其在存檔文件中存儲了上一次產生的ID,由此可保證甚至在有多個軟件實例運行的條件下,都能單調不重復地遞增此ID

  分配給智能指針的tag,唯一地標識出一個指針,而把此tag存入一個存檔文件就是對象串行化過程的責任了。對象的串行化過程,可通過CReferable基類的getTag()方法,來訪問此tag,接下來,對象的反串行化過程使用此tag,在軟件恢復時,來重建正確的對象指針實例引用。下面是反串行化過程必須執行的步驟:

  2 從存檔文件中恢復tag

  2 tag標識的存檔文件中,恢復對象屬性。

  2 以此tag為界調用impersonate()方法,恢復正確的指針對象的引用。

  Impersonate()會對是否一個tag索引了在全局CReferableCache對象集中的一個對象進行檢查,如果未找到此tag相應的對象,那么此對象會添加到CReferableCache中,并用此tag作為它的索引值。然而,如果一個對象已經存在于全局CReferableCache對象集中,通過以新引用來調用set()方法,你可以舍棄老引用,且無關的對象復制操作也會自動被刪除。例2使用了這種技術來實現智能指針。

  例2

以下是引用片段:
classCJobDef:publicCReferable
  {
  friendCArchive&operator<<(CArchive&ar,constCJobDefPtr&cand)
  {
  ar<<cand->getTag();
  CArchivear_def(cand->getTag(),CArchive::WRITE);
  //writeobjectattributestoar_def
  returnar;
  }
  friendCArchive&operator>>(CArchive&ar,CJobDefPtr&cand)
  {
  inttag;
  ar>>tag;
  CArchivear_def(tag,CArchive::READ);
  //readobjectattributesfromar_def
  cand.impersonate(tag);
  returnar;
  }
  };
  classCJobInst:publicCReferable
  {
  friendCArchive&operator<<(CArchive&ar,constCJobInstPtr&cand)
  {
  ar<<cand->m_defPtr;
  returnar;
  }
  friendCArchive&operator>>(CArchive&ar,CJobInstPtr&cand)
  {
  CJobDefPtrdefPtr=newCJobDef;
  ar>>defPtr;
  cand->m_defPtr=defPtr;
  returnar;
  }
  };


 clip_image004

  圖4:作業對象與CReferableCache全局對象的交互

  圖4描述了系統中類CScheduler、CJobDefPtr、CJobDefCReferableCache之間的交互,類CReferableCache具有靜態成員方法getUniqueTag()addObject()、deleteObject()。當一個對CJobDef的智能指針創建時,如下:

以下是引用片段:
CJobDefPtrjobDefPtr=newCJobDef


    CScheduler會構造CJobDefPtr和一個CJobDef對象,當對象構造時,會通過CJobDef基類的CReferable構造函數調用getUniqueTag()方法,這就為每個CJobDef對象創建了一個唯一的識別標記(tag)。接下來,CJobDef對象被賦給CJobDefPtr對象,后者會調用它自己的set()方法把CJobDef對象添加進來。

  當調用CJobDefPtr賦值操作符函數時,也會調用addObject()方法,如果是第一次賦值的話,它會把CJobDef對象添加進全局CReferableCache;當智能指針被請求替換由tag識別的它內部的對象引用時,impersonate()方法會調用getObject()方法,如果impersonate()方法未找到CReferableCache中標記的對象,那么,CJobDefPtr對象會替換它的內部對象標記,并把它自身添加到CReferableCache緩存集中;最后,當CJobDefPtr被刪除及對象的引用計數為零時,deleteObject()方法此時會被調用。

  在此所描述的事件調度系統,一般使用在市場咨詢數據公司中,其會在網絡集群工作站上觸發計算任務,當從世界各處的零售商匯集所需信息之后,在每周的三天之中,都會觸發計算任務,而這三天中的任意時刻,系統可能也要在集群工作站上運行超過20萬個任務。因此,軟件在合理內存及CPU消耗的前提下,支持重新啟動,就顯得非常重要了。表1顯示了在系統中運行著多個計算任務時,事件調度守護進程在每次重啟后的內存消耗,在系統重啟后,較小的內存消耗要歸功于軟件中使用了上文方法來串行及反串行化不常用的類對象的那些模塊。當任務完成時,內存最終將被回收。

  表1:在軟件每次重啟后的調度系統所用內存大小

運行任務數

軟件重啟前的內存占用大小

軟件重啟后的內存占用大小

5000

25M

32M

100000

370M

413M

200000

730M

795M

 

posted on 2009-08-26 13:26 肥仔 閱讀(480) 評論(0)  編輯 收藏 引用 所屬分類: C++ 基礎

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美激情精品久久久久| 久久综合伊人77777尤物| 国产精品专区第二| 欧美手机在线视频| 国产精品日韩高清| 黑人巨大精品欧美一区二区小视频| 国产精品推荐精品| 国产一区二区黄| 亚洲电影免费观看高清完整版在线观看 | 欧美一级电影久久| 久久久九九九九| 老鸭窝亚洲一区二区三区| 欧美韩日一区二区| 国产精品乱码妇女bbbb| 黄色成人在线免费| 一本色道久久综合狠狠躁的推荐| 亚洲桃花岛网站| 久久久蜜桃一区二区人| 91久久精品www人人做人人爽| 亚洲黄网站黄| 亚洲欧美美女| 欧美黄在线观看| 国产日本欧美视频| 亚洲美洲欧洲综合国产一区| 欧美一区激情| 亚洲精品久久久久久久久| 欧美一区二区性| 欧美日韩综合视频网址| 亚洲大片免费看| 欧美在线999| 日韩午夜高潮| 免费观看一区| 激情成人综合| 久久aⅴ国产紧身牛仔裤| 亚洲全部视频| 免费观看成人| 狠狠综合久久| 欧美一区二区播放| 一区二区三区视频观看| 欧美a级在线| 在线观看亚洲| 久久久久成人网| 亚洲欧美日产图| 国产精品久久久久久久久久尿| 亚洲国语精品自产拍在线观看| 国产手机视频一区二区| 欧美视频网址| 亚洲人成网站在线播| 午夜精品免费在线| 日韩视频免费| 久久久久一区二区三区| 国产噜噜噜噜噜久久久久久久久 | 欧美黄色成人网| 精品88久久久久88久久久| 欧美一级精品大片| 亚洲精品免费一二三区| 欧美+日本+国产+在线a∨观看| 韩国成人福利片在线播放| 欧美一区二区三区免费观看| 一本色道**综合亚洲精品蜜桃冫 | 欧美激情亚洲国产| 久久五月天婷婷| 伊人色综合久久天天| 久久久久88色偷偷免费| 欧美怡红院视频一区二区三区| 国产一区二区日韩精品| 久久精品亚洲一区二区| 久久xxxx精品视频| 在线免费观看日本一区| 亚洲第一毛片| 欧美另类变人与禽xxxxx| 99热这里只有成人精品国产| 亚洲精品日韩在线观看| 国产精品乱码| 久久综合伊人77777蜜臀| 久久香蕉国产线看观看av| 亚洲清纯自拍| 国产精品99久久久久久宅男| 国产麻豆91精品| 美腿丝袜亚洲色图| 欧美精品免费在线观看| 亚洲欧美日韩国产成人精品影院| 亚洲女人av| 亚洲国产日本| 亚洲小说春色综合另类电影| 国产亚洲一级| 亚洲人成在线观看网站高清| 国产精品看片资源| 欧美成人a视频| 欧美三级黄美女| 久久久噜噜噜久久狠狠50岁| 欧美大秀在线观看| 欧美一区二区三区免费大片| 久久免费偷拍视频| 亚洲午夜久久久久久久久电影院 | 欧美在线资源| 亚洲麻豆国产自偷在线| 亚洲一区二区三区乱码aⅴ蜜桃女 亚洲一区二区三区乱码aⅴ | 欧美日韩中文字幕日韩欧美| 9国产精品视频| 亚洲一区久久久| 国产综合在线视频| 亚洲黄色尤物视频| 国产色婷婷国产综合在线理论片a| 久久久久久夜| 欧美视频中文字幕在线| 久久综合国产精品| 国产精品久久国产愉拍| 欧美黄色日本| 国产亚洲欧美另类中文 | 91久久极品少妇xxxxⅹ软件| 国产精品一区2区| 亚洲精品美女| 亚洲福利视频三区| 午夜在线观看免费一区| 一区二区av| 欧美91视频| 欧美成年人网| 在线看国产一区| 久久电影一区| 久久国产精品亚洲va麻豆| 欧美视频一区二区三区| 亚洲国产片色| 亚洲激情欧美激情| 久久综合色综合88| 久久综合图片| 激情久久久久| 久久久久久久一区二区| 久久午夜电影| 狠狠色丁香婷婷综合久久片| 小嫩嫩精品导航| 久久精品视频免费| 国产一区二区三区网站| 欧美在线视频在线播放完整版免费观看 | 亚洲国产精品成人综合| 亚洲国产91精品在线观看| 久久久999精品免费| 久久久久久久久久久久久女国产乱| 国产精品日本精品| 亚洲欧美日韩中文播放| 久久精品国产一区二区三| 国产午夜精品美女毛片视频| 欧美一区二区在线看| 久久亚洲综合色一区二区三区| 国产综合在线看| 欧美a级一区| 亚洲最新合集| 久久精品99无色码中文字幕| 国产偷久久久精品专区| 久久久国产精彩视频美女艺术照福利| 久久精品天堂| 亚洲狠狠丁香婷婷综合久久久| 欧美激情2020午夜免费观看| 日韩一级视频免费观看在线| 亚洲一级电影| 国产主播一区二区三区四区| 久久久蜜臀国产一区二区| 亚洲国产高清视频| 亚洲国产欧美日韩另类综合| 亚洲网站在线| 久久久久久久精| 亚洲丶国产丶欧美一区二区三区| 美国十次成人| 91久久精品美女高潮| 一本色道久久综合亚洲二区三区| 国产精品普通话对白| 久久国产精品电影| 亚洲国产精品第一区二区| 亚洲一区二区三区精品视频| 国产一区二区三区奇米久涩| 美女尤物久久精品| 亚洲欧美日韩国产综合| 美女免费视频一区| 亚洲图片激情小说| 伊人婷婷久久| 国产精品女人毛片| 欧美a级一区二区| 亚洲你懂的在线视频| 亚洲国产精品视频一区| 欧美诱惑福利视频| 亚洲美女在线一区| 一区二区三区自拍| 国产精品视频午夜| 免费观看成人网| 欧美在线一级视频| 99精品福利视频| 免费亚洲视频| 久久精品导航| 亚洲欧美国产高清| 亚洲精选中文字幕| 精品不卡视频| 国产乱人伦精品一区二区| 欧美激情一区二区三区在线视频| 欧美一区二区三区四区高清| 亚洲精品一区二| 亚洲成人在线免费| 亚洲综合电影一区二区三区| 91久久综合亚洲鲁鲁五月天| 国产亚洲亚洲| 国产日韩一区二区三区|