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

Benjamin

靜以修身,儉以養(yǎng)德,非澹薄無以明志,非寧靜無以致遠(yuǎn)。
隨筆 - 398, 文章 - 0, 評論 - 196, 引用 - 0
數(shù)據(jù)加載中……

內(nèi)存池和placement new(布局new)

內(nèi)存分配(器)返回的是一塊沒有初始化的內(nèi)存,不返回對象;內(nèi)存分配(器)也不支持設(shè)置成虛指針,因為內(nèi)存分配是在構(gòu)造執(zhí)行后;
1 void* raw = allocate(sizeof(Foo));  // line 1
2  Foo* p = new(raw) Foo();            // line 2 
allocate就是一個簡單的內(nèi)存分配(器),上面的代碼使用了placement new來構(gòu)造對象。
下面一步就是做一個對象的內(nèi)存分配(器)-------------內(nèi)存池(類)
 1 class Pool {
 2  public:
 3    void* alloc(size_t nbytes);
 4    void dealloc(void* p);
 5  private:
 6    data members used in your pool object
 7  };
 8  
 9  void* Pool::alloc(size_t nbytes)
10  {
11    your algorithm goes here
12  }
13  
14  void Pool::dealloc(void* p)
15  {
16    your algorithm goes here
17  } 
分配對象使用下面的代碼
1  Pool pool;
2  
3  void* raw = pool.alloc(sizeof(Foo));
4  Foo* p = new(raw) Foo(); 
或者更簡單的代碼
Foo* p = new(pool.alloc(sizeof(Foo))) Foo(); 
這樣做的好處是可以使用N個內(nèi)存池而不是共享一塊大內(nèi)存,可以從內(nèi)存池中分配而不用釋放,一次釋放整個內(nèi)存池;也可以其中一部分做共享內(nèi)存而不用系統(tǒng)的共享內(nèi)存,從另一個角度來看,很多系統(tǒng)支持alloca()從stack中分配內(nèi)存塊,當(dāng)然這些內(nèi)存塊也會自動離開在函數(shù)返回時,無須delete;當(dāng)然也可以使用alloca()為(內(nèi)存)池分配一個大內(nèi)存塊,而無須調(diào)用delete,在這里析構(gòu)函數(shù)僅僅只是(deallocates)釋放分配的內(nèi)存。
下一步就是就是改變分配對象的語法:與其使用new(pool.alloc(sizeof(Foo))) Foo() 倒不如new(pool) Foo(),為此我們需要實現(xiàn)下面的函數(shù):
 inline void* operator new(size_t nbytes, Pool& pool)
 {
   return pool.alloc(nbytes);
 } 
現(xiàn)在看看這個new(pool)Foo(),它會調(diào)用operator new,通過sizeof(Foo)和pool兩個參量,下面看看如何析構(gòu)對象和釋放內(nèi)存,根據(jù)placement new的析構(gòu)方法,代碼如下:
 void sample(Pool& pool)
 {
   Foo* p = new(pool) Foo();
   
   p->~Foo();        // explicitly call dtor
   pool.dealloc(p);  // explicitly release the memory
 } 
但這樣做,會有下面幾個問題產(chǎn)生:
問題一:防止內(nèi)存泄露,在Foo* p = new Foo(),編譯器會產(chǎn)生類似下面的代碼來應(yīng)對構(gòu)造中的異常
 // This is functionally what happens with Foo* p = new Foo()
 
 Foo* p;
 
 // don't catch exceptions thrown by the allocator itself
 void* raw = operator new(sizeof(Foo));
 
 // catch any exceptions thrown by the ctor
 try {
   p = new(raw) Foo();  // call the ctor with raw as this
 }
 catch () {
   // oops, ctor threw an exception
   operator delete(raw);
   throw;  // rethrow the ctor's exception
 } 
問題是在deallocates(去分配)內(nèi)存時如發(fā)生異常,具體的說就是new的參數(shù)(調(diào)用placement(布局) new)時出現(xiàn)異常,此時編譯器不知所措,默認(rèn)就什么都不做。
 // This is functionally what happens with Foo* p = new(pool) Foo():
 
 void* raw = operator new(sizeof(Foo), pool);
 // the above function simply returns "pool.alloc(sizeof(Foo))"
 
 Foo* p = new(raw) Foo();
 // if the above line "throws", pool.dealloc(raw) is NOT called 
所以編譯器要做一些類似于全局new operator的工作,當(dāng)編譯器看到 new(pool) Foo(),它要查找與之想匹配的delete operator,如果找到了,就相當(dāng)于上面顯示
的包含try的代碼可以正確執(zhí)行,所以要寫個delete operator,像下面的代碼所示(即使第二參數(shù)和operator new(size_t, Pool&)中的第二個參數(shù)不同,也沒有關(guān)系,編譯器也
不會有什么抱怨,此時它會繞過這個try塊)
 void operator delete(void* p, Pool& pool)
 {
   pool.dealloc(p);
 } 
注意:這個operator delete不必一定是inline的,可以是,也可以不是。
destruction(析構(gòu))/deallocation(去分配)是兩個不同的概念,可現(xiàn)實中經(jīng)常混淆這兩個;另外要記住內(nèi)存池里的內(nèi)存和哪個對象匹配,一般我們通過Foo *和
pool* 這兩個指針來定位;對于這兩個方面可能出現(xiàn)的問題,可以用下面的方案來解決。
這個方案就是使每次分配(對象)都關(guān)聯(lián)一個pool *,然后在替代全局delete opeator,如下所示:
void operator delete(void* p)
 {
   if (p != NULL) {
     Pool* pool = /* somehow get the associated 'Pool*' */;
     if (pool == null)
       free(p);
     else
 void* operator new(size_t nbytes)
 {
   if (nbytes == 0)
     nbytes = 1;  // so all alloc's get a distinct address
   void* raw = malloc(nbytes);
   somehow associate the NULL 'Pool*' with 'raw'
   return raw;
 } 

       pool->dealloc(p);
   }
如果不確信dealloc調(diào)用free,可以在全局的new operator中使用malloc(),像下面的代碼所示(此短代碼沒有throw std::bad_alloc()new_handler)
  下面的工作就是用pool*關(guān)聯(lián)每次 allocation(分配),這里我們可以使用STL中的std::map<void*,pool*>,即建立查找的查詢表,其keys是分配指針(void*),value是
pool*,這里要注意,在從全局的operator new向map中添加數(shù)據(jù),因為如果是NULL,會導(dǎo)致遞歸(結(jié)束一次插入同時又插入新項)。
這里用map這個STL里的容器,在大多數(shù)情況是可以接受的。
除了map,還有一種方案就是耗內(nèi)存但是速度快些。就在在pool*之前加上所有的allocate(分配),如果字節(jié)是24,考慮到邊界對齊(double\long long),可能需要28或32字節(jié),首先給pool*四個字節(jié),然后返回4或8個字節(jié)的(指針),從你開始分配時;此時全局的delete opeator至少可以刪除這4或8字節(jié)指針而不會出現(xiàn)異常,這是如果pool*是null,那么用free調(diào)用pool->dealloc(),由free和pool->dealloc()會有4或8字節(jié)的指針到原參p,如果不為空,你就可以決定4字節(jié)對齊,代碼就要像下面示例的這樣:
 void* operator new(size_t nbytes)
 {
   if (nbytes == 0)
     nbytes = 1;                    // so all alloc's get a distinct address
   void* ans = malloc(nbytes + 4);  // overallocate by 4 bytes
   *(Pool**)ans = NULL;             // use NULL in the global new
   return (char*)ans + 4;           // don't let users see the Pool*
 }
 
 void* operator new(size_t nbytes, Pool& pool)
 {
   if (nbytes == 0)
     nbytes = 1;                    // so all alloc's get a distinct address
   void* ans = pool.alloc(nbytes + 4); // overallocate by 4 bytes
   *(Pool**)ans = &pool;            // put the Pool* here
   return (char*)ans + 4;           // don't let users see the Pool*
 }
 
 void operator delete(void* p)
 {
   if (p != NULL) {
     p = (char*)p - 4;              // back off to the Pool*
     Pool* pool = *(Pool**)p;
     if (pool == null)
       free(p);                     // note: 4 bytes left of the original p
     else
       pool->dealloc(p);            // note: 4 bytes left of the original p
   }
 } 
上面的代碼并沒有處理new operator內(nèi)存不足的情況,如果我們不改變?nèi)值膎ew operator和delete operator,上面的問題依然存在。

posted on 2010-05-17 22:35 Benjamin 閱讀(3613) 評論(2)  編輯 收藏 引用 所屬分類: C/C++

評論

# re: 內(nèi)存池和placement new(布局new)  回復(fù)  更多評論   

翻譯的 c++-faq-lite 吧?
2010-05-18 08:58 | 飄飄白云

# re: BREW平臺上如何通過GPS設(shè)備獲取經(jīng)緯度  回復(fù)  更多評論   

高手:BREW平臺上如何通過GPS設(shè)備獲取經(jīng)緯度上的代碼
請問保存為什么格式的文件能夠在手機(jī)上運行啊?感謝
2010-05-20 10:39 | 小雷
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美电影免费观看| 亚洲成色999久久网站| 久久aⅴ国产欧美74aaa| 国产精品久久久久久超碰 | 久久看片网站| 久久视频精品在线| 久久综合久久综合这里只有精品| 久久夜色精品国产欧美乱极品| 女人天堂亚洲aⅴ在线观看| 亚洲第一伊人| 一本色道久久88亚洲综合88| 亚洲女优在线| 欧美1区视频| 国产精品日韩在线| 亚洲电影免费观看高清完整版在线观看 | 国产日韩精品视频一区| 精品成人在线视频| 亚洲精选久久| 欧美中文字幕精品| 亚洲国内精品| 欧美在线看片a免费观看| 欧美成年人在线观看| 国产精品美女视频网站| 一色屋精品视频在线观看网站| 99精品视频免费| 久久久久九九九九| 最新日韩精品| 久久九九免费| 国产精品白丝av嫩草影院| 狠狠色伊人亚洲综合成人| 中文日韩欧美| 蜜桃精品久久久久久久免费影院| 亚洲精品系列| 老司机一区二区三区| 国产精品一页| 亚洲一区二区三区涩| 欧美韩日一区二区三区| 欧美一区二区私人影院日本| 欧美涩涩网站| 91久久综合| 久久久综合香蕉尹人综合网| 亚洲视频一二区| 欧美乱妇高清无乱码| 亚洲成人在线观看视频| 久久久精品国产免大香伊| 亚洲无亚洲人成网站77777 | 亚洲人成小说网站色在线| 久久久国产91| 亚洲欧洲一区二区三区| 久久综合给合久久狠狠色| 99热这里只有成人精品国产| 免费不卡欧美自拍视频| 国精品一区二区| 久久国产婷婷国产香蕉| 亚洲综合久久久久| 国产精品久久久一区麻豆最新章节 | 国产精品久久国产愉拍 | 欧美在现视频| 国产区精品在线观看| 午夜精彩视频在线观看不卡| 一区二区欧美激情| 国产精品嫩草99a| 午夜影院日韩| 欧美一区网站| 激情久久久久| 亚洲第一在线综合网站| 免费中文日韩| 99re热这里只有精品免费视频| 亚洲高清视频中文字幕| 欧美精品日韩三级| 亚洲一区二区毛片| 亚洲自拍偷拍麻豆| 狠狠爱综合网| 亚洲激情在线观看| 国产精品日韩在线播放| 久久一二三国产| 欧美国产精品va在线观看| 一本色道久久综合亚洲精品小说| 日韩小视频在线观看| 国产精品看片资源| 另类激情亚洲| 欧美久久久久| 欧美在线啊v一区| 久久亚洲不卡| 亚洲一区二区四区| 欧美怡红院视频一区二区三区| 1769国产精品| 一本色道久久综合狠狠躁的推荐| 国产精品视频一区二区三区| 蜜乳av另类精品一区二区| 欧美日韩色综合| 久久嫩草精品久久久久| 欧美久久久久免费| 久久精品夜夜夜夜久久| 欧美精品www| 久久美女性网| 欧美午夜欧美| 欧美激情国产日韩| 国产精品视频在线观看| 亚洲第一综合天堂另类专| 欧美午夜寂寞影院| 欧美大片网址| 国产亚洲精品美女| 一区二区日韩伦理片| 国产精品午夜视频| 国产精品国产精品| 免费观看不卡av| 国产精品美女主播在线观看纯欲| 久久久亚洲国产天美传媒修理工| 欧美日韩a区| 欧美国产高清| 国产视频一区在线| 一本一本久久a久久精品牛牛影视| 亚洲成人直播| 亚洲性感美女99在线| 久久成人亚洲| 亚洲一区国产视频| 欧美成人福利视频| 毛片一区二区三区| 国产亚洲欧美一级| 亚洲一区二区三区中文字幕| 亚洲精品一二三| 久久婷婷蜜乳一本欲蜜臀| 亚洲欧美中文在线视频| 欧美日韩国产在线播放| 欧美国产精品中文字幕| 一区三区视频| 久热精品视频在线免费观看| 久久在线免费观看视频| 国产日韩欧美在线播放不卡| 亚洲女人天堂av| 午夜亚洲伦理| 国产毛片精品国产一区二区三区| 一区二区国产在线观看| 亚洲图片在线| 国产精品国产一区二区| 这里只有精品在线播放| 一区二区三区视频在线播放| 欧美精品国产| 一区二区三区国产在线| 中国女人久久久| 国产精品videosex极品| 一区二区福利| 欧美在线综合| 黄色一区二区三区| 久久综合九色综合久99| 欧美激情视频在线免费观看 欧美视频免费一 | 国产精品久久久久久久午夜片| 一区二区三区四区五区视频| 亚洲午夜视频在线观看| 国产精品视频一二三| 欧美一区二区三区啪啪| 噜噜噜躁狠狠躁狠狠精品视频| 一区在线免费| 欧美日韩高清在线观看| 中文精品视频一区二区在线观看| 午夜精品久久一牛影视| 国产亚洲一区在线| 农村妇女精品| 一本色道久久综合亚洲精品小说| 午夜亚洲精品| 亚洲国产日韩在线| 欧美午夜精品久久久久久久| 欧美一区二区视频免费观看 | 欧美亚洲日本网站| 美腿丝袜亚洲色图| 一区二区三区国产盗摄| 国产日韩精品一区观看| 亚洲天堂男人| 午夜精品免费| 狂野欧美一区| 亚洲免费观看高清完整版在线观看熊| 欧美激情二区三区| 亚洲一区久久久| 男人插女人欧美| 亚洲自啪免费| 亚洲黄色免费电影| 国产欧美亚洲一区| 欧美大片一区二区三区| 亚洲欧美视频一区| 亚洲黄色一区| 久久久久久噜噜噜久久久精品| 亚洲毛片在线观看| 狠狠色狠狠色综合系列| 欧美性做爰猛烈叫床潮| 老牛嫩草一区二区三区日本| 亚洲网站视频| 亚洲美女毛片| 欧美成人精品h版在线观看| 午夜电影亚洲| av成人手机在线| 在线观看亚洲视频| 国产日本亚洲高清| 国产精品电影观看| 欧美美女日韩| 老司机午夜精品视频在线观看| 午夜精品美女自拍福到在线| 夜夜爽av福利精品导航| 亚洲国产乱码最新视频| 免费观看成人|