在程序設計中頻繁地進行動態內存分配和釋放容易造成內存碎片,給需要長期穩定運行的系統帶來了隱患。盡管現代操作系統在內存管理的穩健性上已經有了較大提高,但是動態內存分配的效率、穩定性等問題仍然是一個困擾。所以在需要長期穩定運行的、生命攸關的嵌入式設備程序開發中通常都要避免過多地使用new和delete,甚至禁止使用C++的標準庫,因為其中經常進行動態內存的分配與釋放。然而要充分享用使用C++帶來的便利之處,動態地創建和銷毀對象的能力還是非常有必要的。
本文引用地址://www.einuk.cn/emb/Column/7365.html
通常的new運算符會在堆中動態分配內存,并在成功分配的內存空間上調用相應對象的構造函數,而delete運算符會先在該內存空間上調用相應對象的析構函數,然后釋放該內存空間。然而在C++中,可以通過對new和delete的重載來實現特殊的功能。不過在程序員編寫的new和delete的重載函數中都只是負責內存空間的分配和釋放,編譯器會在new時強制調用構造函數,在delete時強制調用析構函數。因為在C++中初始化和清除工作是強制進行的,用戶無權選擇避開構造函數和析構函數的調用。
要想在這些特殊的情況下能實現對象的創建和銷毀,C++中提供了placement new來應對。在C++中因為可以對運算符進行重載,所以只需對operator new()進行placement重載就能實現在事先分配好的內存空間上調用構造函數來動態創建對象,通過不太常用的對析構函數的顯式調用來銷毀對象。在這種情況下不涉及內存空間的分配和釋放,從而不會導致動態的存儲管理,不會對系統的長期穩定運行帶來影響。
在C++中placement new運算符是通過對new運算符的重載來實現的,它在嵌入式開發中有著重要的用途。因為我們經常想在內存的某個指定位置上放置一個對象(在嵌入式底層開發中,一個對象可能和一個特定的硬件是直接相關的)。
Placement new的主要用途就是:反復使用一塊較大的分配成功的內存來構造不同類型的對象或者它們的數組。比如,可以先申請一個足夠大的字符數組,然后當需要時在它上面構造不同類型的對象或其數組。
定位new運算符的基本語法為
char a[100];
X* xp = new(a) X;
其中X是某個事先定義好類,a是一塊尺寸大于等于sizeof(X)的內存區域的首地址。由于內存區域是事先已經存在的,placement僅僅只是在相應的內存空間上調用X的構造函數來初始化該片內存區域。
下面的例子顯示了如何在一個特定的內存單元里放置一個對象。
#include
using namespace std;
class X {
int i;
public:
X(int ii = 0) : i(ii) {
cout << "this = " << this << endl;
}
~X() {
cout << "X::~X(): " << this << endl;
}
void* operator new(size_t, void* loc) {
return loc;
}
};
int main()
{
int l[10];
cout << "l = " << l << endl;
X* xp = new(l) X(47);
xp->X::~X();
return 0;
}
從中可以看到數組l的地址和構造以及析構函數中輸出的this指針的值是相同的。
注意operator new()僅僅返回了傳遞給它的指針,因此調用者可以決定將對象存放在哪里,這時在該指針所指向的那塊內存上,作new表達式一部分的構造函數將被調用。概括來說就是內存空間是事先分配好的,placement new只是在其上調用構造函數來動態創建對象。
然而在銷毀對象時則需要格外小心,因為空間不是由new創建的,所以在銷毀對象時也應該將對象的析構和內存空間的釋放分開來做。我們要銷毀對象就需要調用析構函數,但此時不能將內存空間釋放掉,因為內存空間不是由new所分配出來的。解決辦法是用非常特殊的語法,我們可以顯式的調用構造函數。例如:
xp->X::~X();
這樣將會只銷毀對象而不會釋放內存。通過placement new和顯式調用析構函數的方式我們就可以在已有的空間上動態地創建和銷毀對象。在嵌入式領域,為了穩定性,我們可以在程序一開始時分配好足夠的空間,在上面構造和銷毀對象,充分享受這種便利性,同時也避免了通常的new和delete所帶來的內存碎片問題。
使用placement new構造起來的對象或其數組,要顯示地調用它們的析構函數來銷毀(析構函數并不釋放對象的內存),千萬不要使用delete。這是因為,placement new構造起來的對象或其數組的大小并不一定等于原來分配的內存大小,而且空間也不一定是在堆上分配的,使用delete的話將會出現嚴重的問題。
华清图书馆
0元电子书,限时免费申领10本华清图书PDF版
扫码关注华清远见公众号