定位new 運算符

定位new  

2007-10-08 18:03:02|  分類: 技術總結|舉報|字號 訂閱

在預先定義的內存位置構造一個對象

    常常有人問這樣一個C++問題:如何在預先定義的內存位置構造一個對象?在預先定義的內存緩衝構造一個對象有許多有用的應用。例如,一個定製的垃圾蒐集器能使用一個大的預
分配內存緩衝,用戶在這個緩衝中構造其對象。當不再需要這些對象時,它們的存儲空間被自動收回。

   這個技術在重視時間的應用中也很有用。在預先分配的內存緩衝構造一個對象是一種“時間常量”操作,之所以這樣說是因爲程序分配操作本身不會浪費寶貴的時間。同時也要注意當系統沒有足夠的內存時,動態內存分配可能失敗。因此,對於重視任務的應用,預先分配一個足夠大的緩衝有時是不可避免的。

    許多應用需要在給定的時間構造不同類型的對象。想一想這樣一個例子,一個GUI應用根據用戶的輸入,每次、顯示不同的對話框,利用重複分配和釋放內存,這個應用能提前創建一個內存緩衝,並能在這個緩衝裏反覆構造和銷燬不同類型的對象。

    C++提供了幾種特點來方便實現在預先決定的內存位置構造一個對象的任務。在這些特點中,包括一個特殊形式的new操作符,叫做“定位new”(placement new)操作,以及一個顯式的析構處理。實現方法如下:

    第一步:分配一個足夠的內存緩衝區,以便存放給定類型的對象。如果想要每次構造不同類型的對象,需要至少以最大的對象所佔空間的大小分配一個緩衝。預分配的緩衝是在可用內存空間中分配的純字符數組。

char * buff = new char [sizeof (Foo) ];

    一旦分配了緩衝,就能在緩衝中構造每一種類型的對象。爲此,使用特殊版本的new操作符(“定位new”),以緩衝地址爲placement new的參數。爲了使用placement new,必須包含標準頭文件<new>。下面的代碼片斷中,使用placement new操作在內存地址buff上構造類型爲Foo的對象。

#include <new>
Foo * pfoo = new (buff) Foo; //使用new操作在buff上構造一個 Foo 

    Placement new 以先前分配的緩衝(buff)地址作爲參數,並在這個緩衝上構造給定類型的對象。他返回構造對象的指針,這個對象指針的使用與通常的指針使用沒什麼兩樣。

unsigned int length = pfoo->size(); 
pfoo->resize(100, 200);
length = pfoo->size(); 

    當不再需要這個對象的時候,必須顯式調用其析構函數釋放空間。做這件事是有一些技巧的,因爲許多人錯誤地假設對象會被自動銷燬,錯也!。在預分配的緩衝裏構造另一個對象之前或者在釋放緩衝之前,如果忘了顯式調用析構函數,程序將產生不可預料的後果。顯式的析構器聲明如下:

pfoo->~Foo(); //顯式調用析構函數

    換句話說,一個顯式的析構器與普通的成員函數調用一樣,只是名字與普通的成員函數稍有差別。一旦對象被銷燬,便可以在預分配的內存中再次構造另一個對象。實際上,這個過程可以無限制地重複:構造一個對象,銷燬它,然後又反覆利用預分配的緩衝構造新對象。

    當不再需要預定義的緩衝時,或者說當應用程序關閉時,必須釋放預定義的緩衝。使用delete[]完成這個任務,因爲預定義的緩衝是一個字符數組。下列代碼包含一個完整的例子的所有步驟,包括最終緩衝的釋放:
#include <new>

  void placement_demo()
  { 
    //1. 預分配緩衝
    char * buff = new char [sizeof (Foo) ];  

    //2. 使用 placement new
    Foo * pfoo = new (buff) Foo;  
    
    //使用對象
    unsigned int length = pfoo->size();  
    pfoo->resize(100, 200);

    //3. 顯式調用析構函數
    pfoo->~Foo();  
    
    //4. 釋放預定義的緩衝
    delete [] buff;  
  }
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章