基本元素
new
operator new
placement new
array new
一.new和operator new
1.new expression 的操作偽代碼
complex * pc = new complex(1,2)
try{
void * mem = operator new(sizeof (complex)); // operator new可以被重載,沒有重載的話缠诅,調(diào)用全局的::operator new
pc = static_cast<complex*>(mem);
pc->complex::complex(1,2); //直接調(diào)用 構(gòu)造函數(shù)乍迄,部分編譯器才可以做的操作
//或者調(diào)用 new(p) Complex(1,2) placement new
}
catch(std::bad_alloc){
}
2.operator new的全局版本 (vc版本)
operator new (size_t size,std::nothrow_t &) _THROW0{
void *P;
while((p = malloc(size) == 0){
_TRY_BEGIN
if(callnewh(size) == 0 ) break; //callnweh由用戶自己設(shè)定,可能可以釋放一些覺得不重要的內(nèi)存這樣在下次malloc中可能可以成功
_CATCH(astd::bad_alloc) return(0);
_CATCH_END
})
}
- 第一個參數(shù)“必須”是size_t褥伴,以接受new expression的以sizeof(T)為實參的調(diào)用
- callnewh(size) 由用戶自己設(shè)定,可能可以釋放一些覺得不重要的內(nèi)存這樣在下次malloc中可能可以成功
- std::nothrow_t & 保證這個函數(shù)不拋出異常漾狼,指明這個函數(shù)通過返回一個空指針來表明分配失敗重慢,而不是拋出異常
3.delete expression
pc-> ~ Complex();
operator delete(pc);
4.array new
int * pi = new int[10] 的內(nèi)存空間 (默認(rèn)行為下的內(nèi)存空間)
- delete[] 會調(diào)用多次析構(gòu)函數(shù),如果對一個管理堆內(nèi)存的對象來說會有影響逊躁。
- delete 把指針指向的位置當(dāng)成一個對象去解析似踱,只會調(diào)用一次析構(gòu)函數(shù),所以并不會造成內(nèi)存泄漏
- 必須要有默認(rèn)構(gòu)造函數(shù)稽煤,array沒機會給掉帶參數(shù)的
- new[]從地地址向高地址new, delete[] 從高地址向低地址delete
- Debugger Header 在debug模式下出現(xiàn)
- 頭和尾有個cookie核芽,記錄整個分配空間的大小(malloc行為,每一次都會有)
- 內(nèi)存要16字節(jié)對齊酵熙,所以添加了 12個字節(jié)的填補
二.placement new
1.placement new
處理流程和new expression一樣
區(qū)別在于對于operator new的調(diào)用上
string s;
string* s= new(&s) string("a");
調(diào)用void* operator new(size_t,void * start) { return start ;}
string s;
string* s= new(300) string("a");
調(diào)用void* operator new(size_t len,size_t extra) { malloc(len+extra) ;}
- palcement new中調(diào)用operator new時轧简,第一個參數(shù)是sizeof的值,之后再傳遞placement new接受的參數(shù)
1.placement delete的重載
placement delete的重載可以被自動調(diào)用匾二。
每一個new操作都有分配內(nèi)存,類型轉(zhuǎn)換察藐,調(diào)用構(gòu)造三步操作
如果在調(diào)用構(gòu)造時拋出異常皮璧,為了不讓之前的分配出的內(nèi)存泄漏,會自動調(diào)用與之對應(yīng)的operator new的重載形式
operator new(size_size ,long) 會自動調(diào)用 void operator delete(void * ,long)
operator new(size_size ) 會自動調(diào)用 void operator delete(void *)
如果沒有找到對應(yīng)的重載分飞, 則被視為放棄處理