日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

c++中的new、operator new、placement new

發布時間:2025/3/14 c/c++ 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 c++中的new、operator new、placement new 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、定義

1、new

new是c++中的關鍵字,,其行為總是一致的。它先調用operator new分配內存,然后調用構造函數初始化那段內存。

new 操作符的執行過程:
1. 調用operator new分配內存 ;
2. 調用構造函數在operator new返回的內存地址處生成類對象;

2、operator new

operator new是一個函數,就像重載任何一個符號如operator +,它用來分配內存(只不過new除了調用它還有其他步驟)。它可以被重載,通過重載它,可以改變new操作符的功能。它的功能介意類比c語言中的malloc,如果類中沒有重載operator new,那么調用的就是全局的::operator new來從堆中分配內存。

2、placement new

placement new 是c++中對operator new 的一個標準、全局的重載版本。它并不分配內存,只是返回指向已經分配好的某段內存的一個指針,placement new允許你在一個已經分配好的內存中(棧或者堆中)構造一個新的對象。

二、使用方法

1、new的使用

在堆上分配分配一塊內存

struct A* i0 = new A; struct A* i1 = new A();

看new的原型:

void* operator new(std::size_t) _GLIBCXX_THROW (std::bad_alloc)__attribute__((__externally_visible__)); void* operator new[](std::size_t) _GLIBCXX_THROW (std::bad_alloc)__attribute__((__externally_visible__)); void operator delete(void*) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__)); void operator delete[](void*) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__)); void* operator new(std::size_t, const std::nothrow_t&) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__)); void* operator new[](std::size_t, const std::nothrow_t&) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__)); void operator delete(void*, const std::nothrow_t&) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__)); void operator delete[](void*, const std::nothrow_t&) _GLIBCXX_USE_NOEXCEPT__attribute__((__externally_visible__));

發現它有一個參數,size_t,表示前面調用placement new分配的內存大小,new接下來會在這塊內存中調用構造函數,new的操作也是c++來保證的。

2、operator new的使用

分配8個字節的內存,因為是未重載,所以這里調用的是全局operator new,從堆上分配了8個字節

void* i = operator new (8);

如果想重載operator new需要注意以下幾點:

(1)重載時,返回類型必須聲明為void*

(2)重載時,第一個參數類型必須為表達要求分配空間的大小(字節),類型為size_t

(3)重載時,可以帶其它參數

(4)分配函數為類成員函數或全局函數;如果分配函數在全局范圍之外的名稱空間范圍中聲明,或者在全局范圍中聲明為靜態,則程序是病態的

比如下面的例子中,在A重載了operator new打印出tag,返回全局的opereator new,然后在main函數中調用A的重載版本。

struct A{int a;char b;void* operator new(size_t size,int tag) throw(){cout << tag << endl;return ::operator new(size);} };int main() {void* i = A::operator new (8,1);cout << i << endl;return 0; }

最終結果即分配了內存,又打印出了tag的值

如果我們重載全局的operator new函數,然后調用new,則new的操作也會被更改,比如下面的例子(這個例子的operator new只有一個參數)

struct A{int a;char b; };
void* operator new(size_t num) throw(){cout << num << endl;return nullptr; }int main() {A* i = new A;cout << i << endl;return 0; }

最終的結果是

可以看出,雖然沒有直接調用operator new,但是new的操作已經被更改了。

還需要關注一個小地方,就是operator new調用時的參數和new的參數是有所區別的。new在調用的時候會忽略第一個size_t的參數,但是如果直接調用operator new來進行內存分配的時候是需要這個參數的。

也就是本節的第二個例子如果operator new的定義要像本節的第一個例子有兩個參數的話,對new的調用應該如下:

A* i = new(1) A ;

3、placement new的使用

placement new是c++實現的operator new版本,它的實現如下

// Default placement versions of operator new. inline void* operator new(std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT { return __p; } inline void* operator new[](std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT { return __p; }// Default placement versions of operator delete. inline void operator delete (void*, void*) _GLIBCXX_USE_NOEXCEPT { } inline void operator delete[](void*, void*) _GLIBCXX_USE_NOEXCEPT { } //@}

可以看到實際上它就返回了傳進來的地址,根據operator的第二個例子,通過重載全局的operator new之后,new函數的操作就被改變了。也就能猜出,在調用new的時候參數需要加上一個地址,placement new的功能就是在這個地址之上進行構造。

placement new的使用步驟如下

1)分配內存

char* buff = new char[ sizeof(Foo) * N ]; memset( buff, 0, sizeof(Foo)*N );

2)構建對象

Foo* pfoo = new (buff)Foo;

3)使用對象

pfoo->print(); pfoo->set_f(1.0f); pfoo->get_f();

4)析構對象,顯式的調用類的析構函數。

pfoo->~Foo();

5)銷毀內存

delete [] buff;

上面5個步驟是標準的placement new的使用方法。

?

轉載于:https://www.cnblogs.com/likaiming/p/9393083.html

與50位技術專家面對面20年技術見證,附贈技術全景圖

總結

以上是生活随笔為你收集整理的c++中的new、operator new、placement new的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。