C++中的new其實是一個很糊弄人的術語,它有兩種不同的含義,new運算符(new operator)和new函數(operator new),值得記錄一下。
一 new運算符
最常用的是作為運算符的new,比如:
string *str = new string(“test new”);
作為運算符,new和sizeof一樣,是C++內置的,你不能對它做任何的改變,除了使用它。
new會在堆上分配一塊內存,並會自動調用類的構造函數。
二 new函數
第二種就是new函數,其實new運算符內部分配內存使用的就是new函數,原型是:
void *operator new(size_t size);
new函數返回的是一個void指針,一塊未經初始化的內存。如你所見,這和C語言的malloc行為相似,你可以重載new函數,並且增加額外的參數,但是必須保證第一個參數必須是size_t類型,它指明了分配內存塊的大小,C++允許你這麼做,當然一般情況下這是不必要的。如果重載了new函數,在使用new操作符時調用的就是你重載後的new函數了。
如果使用new函數,和語句string *str = new string(“test new”)相對的代碼大概是如下的樣子:
string *str = (string*)operator new(sizeof(string));
str.string(“test new”); // 當然這個調用時非法的,但是編譯器是沒有這個限制的
這還不算完,還有第三種的new存在。
三 placement new
第三種,placement new,這也是new作為函數的一種用法,它允許你在一塊已存在的內存上分配一個對象,而內存上的數據不會被覆蓋或者被你主動改寫,placement new同樣由new操作符調用,調用格式是:
new (buffer) type(size_t size);
先看看下面的代碼:
char str[22];
int data = 123;
int *pa = new (&data) int;
int *pb = new (str) int(9);
結果*pa = 123(未覆蓋原數據),而*pb = 9(覆蓋原數據),可以看到placement new 並沒有分配新的內存,也可以使用在棧上分配的內存,而不限於堆。
為了使用placement new 你必須包含<new>或者<new.h>
其實placement new和第二種一樣,只不過多了參數,是函數new的重載,語法格式為:
void *operator new(size_t, void* buffer);
它看起來可能是這個樣子:
void *operator new(size_t, void* buffer) { return buffer;}
和new對應的就是delete了,需要回收內存啊,不然就洩漏了,這個下次再寫吧,回憶一下今天的內容先。
總結
1. 函數new
void *operator new(size_t size); 在堆上分配一塊內存,和placement new(void *operator new(size_t, void* buffer)); 在一塊已經存在的內存上創建對象,如果你已經有一塊內存,placement new會非常有用,事實上,它STL中有著廣泛的使用。
2. 運算符new
最常用的new,沒什麼可說的。
3. 函數new不會自動調用類的構造函數,因為它對分配的內存類型一無所知;而運算符new會自動調用類的構造函數。
4. 函數new允許重載,而運算符new不能被重載。
5. 緊接著就是對應的delete。