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

Benjamin

靜以修身,儉以養德,非澹薄無以明志,非寧靜無以致遠。
隨筆 - 398, 文章 - 0, 評論 - 196, 引用 - 0
數據加載中……

內存池和placement new(布局new)

內存分配(器)返回的是一塊沒有初始化的內存,不返回對象;內存分配(器)也不支持設置成虛指針,因為內存分配是在構造執行后;
1 void* raw = allocate(sizeof(Foo));  // line 1
2  Foo* p = new(raw) Foo();            // line 2 
allocate就是一個簡單的內存分配(器),上面的代碼使用了placement new來構造對象。
下面一步就是做一個對象的內存分配(器)-------------內存池(類)
 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個內存池而不是共享一塊大內存,可以從內存池中分配而不用釋放,一次釋放整個內存池;也可以其中一部分做共享內存而不用系統的共享內存,從另一個角度來看,很多系統支持alloca()從stack中分配內存塊,當然這些內存塊也會自動離開在函數返回時,無須delete;當然也可以使用alloca()為(內存)池分配一個大內存塊,而無須調用delete,在這里析構函數僅僅只是(deallocates)釋放分配的內存。
下一步就是就是改變分配對象的語法:與其使用new(pool.alloc(sizeof(Foo))) Foo() 倒不如new(pool) Foo(),為此我們需要實現下面的函數:
 inline void* operator new(size_t nbytes, Pool& pool)
 {
   return pool.alloc(nbytes);
 } 
現在看看這個new(pool)Foo(),它會調用operator new,通過sizeof(Foo)和pool兩個參量,下面看看如何析構對象和釋放內存,根據placement new的析構方法,代碼如下:
 void sample(Pool& pool)
 {
   Foo* p = new(pool) Foo();
   
   p->~Foo();        // explicitly call dtor
   pool.dealloc(p);  // explicitly release the memory
 } 
但這樣做,會有下面幾個問題產生:
問題一:防止內存泄露,在Foo* p = new Foo(),編譯器會產生類似下面的代碼來應對構造中的異常
 // 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(去分配)內存時如發生異常,具體的說就是new的參數(調用placement(布局) new)時出現異常,此時編譯器不知所措,默認就什么都不做。
 // 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的工作,當編譯器看到 new(pool) Foo(),它要查找與之想匹配的delete operator,如果找到了,就相當于上面顯示
的包含try的代碼可以正確執行,所以要寫個delete operator,像下面的代碼所示(即使第二參數和operator new(size_t, Pool&)中的第二個參數不同,也沒有關系,編譯器也
不會有什么抱怨,此時它會繞過這個try塊)
 void operator delete(void* p, Pool& pool)
 {
   pool.dealloc(p);
 } 
注意:這個operator delete不必一定是inline的,可以是,也可以不是。
destruction(析構)/deallocation(去分配)是兩個不同的概念,可現實中經常混淆這兩個;另外要記住內存池里的內存和哪個對象匹配,一般我們通過Foo *和
pool* 這兩個指針來定位;對于這兩個方面可能出現的問題,可以用下面的方案來解決。
這個方案就是使每次分配(對象)都關聯一個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調用free,可以在全局的new operator中使用malloc(),像下面的代碼所示(此短代碼沒有throw std::bad_alloc()new_handler)
  下面的工作就是用pool*關聯每次 allocation(分配),這里我們可以使用STL中的std::map<void*,pool*>,即建立查找的查詢表,其keys是分配指針(void*),value是
pool*,這里要注意,在從全局的operator new向map中添加數據,因為如果是NULL,會導致遞歸(結束一次插入同時又插入新項)。
這里用map這個STL里的容器,在大多數情況是可以接受的。
除了map,還有一種方案就是耗內存但是速度快些。就在在pool*之前加上所有的allocate(分配),如果字節是24,考慮到邊界對齊(double\long long),可能需要28或32字節,首先給pool*四個字節,然后返回4或8個字節的(指針),從你開始分配時;此時全局的delete opeator至少可以刪除這4或8字節指針而不會出現異常,這是如果pool*是null,那么用free調用pool->dealloc(),由free和pool->dealloc()會有4或8字節的指針到原參p,如果不為空,你就可以決定4字節對齊,代碼就要像下面示例的這樣:
 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內存不足的情況,如果我們不改變全局的new operator和delete operator,上面的問題依然存在。

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

評論

# re: 內存池和placement new(布局new)  回復  更多評論   

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

# re: BREW平臺上如何通過GPS設備獲取經緯度  回復  更多評論   

高手:BREW平臺上如何通過GPS設備獲取經緯度上的代碼
請問保存為什么格式的文件能夠在手機上運行啊?感謝
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>
            久久视频这里只有精品| 亚洲午夜激情网页| 欧美激情亚洲| 欧美制服丝袜| 亚洲精品字幕| 亚洲国产视频直播| 尤妮丝一区二区裸体视频| 一区二区三区精品视频| 亚洲激情第一页| 欧美高清视频在线| 浪潮色综合久久天堂| 性做久久久久久免费观看欧美| 1769国产精品| 在线观看国产日韩| 在线欧美日韩国产| 亚洲国产精品一区二区第四页av| 影音先锋在线一区| 狠狠网亚洲精品| 好吊成人免视频| 在线免费观看欧美| 夜夜夜精品看看| 亚洲一级片在线看| 久久成人一区二区| 美女精品在线| 亚洲一区二区三区免费视频 | 欧美精品乱人伦久久久久久| 久久久久久久久久看片| 久久国产88| 老司机免费视频久久| 能在线观看的日韩av| 欧美国产综合一区二区| 欧美日本国产| 国产精品成人aaaaa网站| 欧美视频二区| 国产精品日日摸夜夜添夜夜av| 国产精品久久久久久模特| 国产精品亚洲综合一区在线观看| 国产私拍一区| 亚洲黄网站黄| 一本一本久久a久久精品综合麻豆| 一区二区不卡在线视频 午夜欧美不卡'| 亚洲美女视频| 午夜亚洲性色福利视频| 久久精品中文字幕一区| 欧美高清在线一区| 亚洲午夜小视频| 亚洲欧美国产视频| 美腿丝袜亚洲色图| 国产农村妇女毛片精品久久麻豆| 国产日韩一区二区三区在线播放| 国产性色一区二区| 亚洲片区在线| 欧美亚洲一区二区在线观看| 久久婷婷成人综合色| 亚洲黄色性网站| 亚洲免费在线视频| 欧美成人高清视频| 国产欧美一区二区三区国产幕精品| 狠狠综合久久av一区二区小说| 国产日韩欧美电影在线观看| 亚洲精品乱码久久久久久蜜桃麻豆 | 一区二区激情| 久久精品理论片| 亚洲国产你懂的| 久久久久久久精| 国产精品狼人久久影院观看方式| 在线看国产日韩| 久久成年人视频| a4yy欧美一区二区三区| 免费久久精品视频| 国产精品久久国产三级国电话系列 | 国产有码在线一区二区视频| 亚洲视频播放| 欧美国产综合一区二区| 欧美伊人久久| 国产日韩精品一区二区| 国产精品99久久久久久人 | 欧美色中文字幕| 99国产精品一区| 亚洲国产激情| 欧美中文在线免费| 国产精品国产三级国产专播精品人 | 久久久久久电影| 亚洲图片在线观看| 欧美成年人视频| 亚洲精品欧美日韩| 亚洲国产欧美不卡在线观看| 中文国产成人精品| 亚洲精品国久久99热| 免费看精品久久片| 亚洲人成网在线播放| 欧美v日韩v国产v| 午夜久久美女| 国产日韩欧美中文| 久久久久久高潮国产精品视| 久久国产精品久久w女人spa| 国产一区二区看久久| 久久精品国产77777蜜臀| 欧美在线网站| 伊甸园精品99久久久久久| 久久久久成人网| 久久亚洲精品伦理| 国产精品网曝门| 欧美亚州韩日在线看免费版国语版| 一区二区三区欧美日韩| 一本色道久久加勒比88综合| 国产精品高潮在线| 亚洲免费大片| 亚洲影视在线播放| 狠狠综合久久| 亚洲精品欧美精品| 国产麻豆一精品一av一免费| 久久在线观看视频| 欧美黑人国产人伦爽爽爽| 中国亚洲黄色| 久久国产精品72免费观看| 亚洲欧洲在线观看| 亚洲国产综合在线看不卡| 一二三区精品| 欧美日韩亚洲一区二区三区在线观看 | 欧美性天天影院| 久久婷婷人人澡人人喊人人爽| 久热精品视频| 999在线观看精品免费不卡网站| 亚洲午夜电影| 亚洲国产你懂的| 亚洲天堂视频在线观看| 国内精品视频一区| 欧美福利视频| 欧美专区在线观看| 最新亚洲激情| 欧美在线视频免费| 亚洲一区综合| 美女亚洲精品| 欧美综合国产精品久久丁香| 免费成人小视频| 久久电影一区| 久久久免费精品| 欧美一区在线看| 欧美激情综合五月色丁香小说 | 欧美激情综合| 免费日韩av| 国产亚洲欧美色| 99re8这里有精品热视频免费| 影音先锋欧美精品| 亚洲免费视频一区二区| 一区二区日韩精品| 女人天堂亚洲aⅴ在线观看| 欧美在线观看视频一区二区三区| 欧美成人精品一区二区| 亚洲欧洲av一区二区| 久久国产婷婷国产香蕉| 亚洲在线一区二区三区| 欧美日韩国产成人在线91| 亚洲第一精品电影| 亚洲第一页自拍| 久久久青草青青国产亚洲免观| 欧美一区免费视频| 欧美二区乱c少妇| 亚洲综合国产| 欧美日韩精品不卡| 99成人在线| 夜夜嗨网站十八久久| 久久国产欧美日韩精品| 在线日本欧美| 久久一区激情| 亚洲福利在线看| 亚洲精品资源| 免费在线看成人av| 久久影院午夜片一区| 久久精品中文字幕免费mv| 在线播放中文字幕一区| 久久久久久久网站| 亚洲激情视频在线观看| 99re6这里只有精品视频在线观看| 欧美日韩国产天堂| 99在线热播精品免费99热| 久久精品官网| 欧美午夜免费影院| 久久精品人人做人人综合| 91久久精品日日躁夜夜躁国产| 这里只有精品视频在线| 国产综合激情| 欧美精品成人91久久久久久久| 亚洲一区免费网站| 久久蜜臀精品av| 亚洲欧美变态国产另类| 国产精品私拍pans大尺度在线| 欧美亚洲视频在线看网址| 欧美成人亚洲成人| 在线亚洲激情| 国内精品美女在线观看| 欧美激情精品久久久久久大尺度 | 亚洲高清久久久| 欧美日本国产一区| 欧美一区二区三区在线观看| 日韩图片一区| 久久久91精品| 欧美一区二区免费| 国产在线成人|