1. 程式人生 > >malloc和new

malloc和new

1.malloc

int *p = (int *)malloc(sizeof(int))

1.calloc

void *calloc( size_t num, size_t size );

函式返回一個指向num 陣列空間,每一陣列元素的大小為size。如果錯誤發生返回NULL。

2.malloc

void *malloc( size_t size );

函式指向一個大小為size的空間,如果錯誤發生返回NULL。

3.realloc

void *realloc( void *ptr, size_t size )

函式將ptr 物件的儲存空間改變為給定的大小size。 返回值是指向新空間的指標,如果錯誤發生返回NULL。

4.總結:

1.函式使用需包含一個頭檔案#include<stdlib.h>

2.malloc函式會返回開闢空間的首地址

3.由malloc動態申請的記憶體空間是堆式的記憶體空間。而靜態的記憶體的空間是棧式的

4.當需要的記憶體過大,或者不確定記憶體需求時,則使用動態記憶體開闢空間

2.new

1.new operator(new表示式)

class A{

    int a;

};

A * array=new A[101];

申請記憶體和初始化

2.operator new (new操作符)

類似於malloc,只負責申請記憶體

A* array=(A*) ::operator new(sizeof(A) *101);

3.placement new

用於在給定的記憶體中初始化物件

class A{
    int a;
    void* operator new(size_t size)
    printf("haha\n");
}

注意:

1、使用者是無法主動呼叫建構函式的,所以需要藉助placement new,但是使用者可以主動呼叫解構函式,所以用完這些物件後,呼叫解構函式,然後用對應分配記憶體的方法去釋放記憶體。

2、事實上malloc並不一定比operatornew節省多少時間,用placement new常常是為了考慮效能,所以會配合記憶體池一起使用。

4.總結

1.new int;//開闢一個存放整數的儲存空間,返回一個指向該儲存空間的地址(即指標)

2.new int(100);//開闢一個存放整數的空間,並指定該整數的初值為100,返回一個指向該儲存空間的地址

3.new int[5][4];//開闢一個存放二維整型陣列(大小為5*4)的空間,返回首元素的地址。用new分配陣列空間時不能指定初值。

4.delete [] pt;//在指標變數前面加一對方括號,表示是對陣列空間的操作

3.malloc/free 和new/delete的區別

1.屬性:

new/delete是C++關鍵字,需要編譯器支援。malloc/free是C語言庫函式,需要標頭檔案支援c。

2.記憶體區域:

new操作符從自由儲存區(free store)上為物件動態分配記憶體空間,而malloc函式從堆上動態分配記憶體。(自由儲存區是C++基於new操作符的一個抽象概念,凡是通過new操作符進行記憶體申請,該記憶體即為自由儲存區。而堆是作業系統中的術語,是作業系統所維護的一塊特殊記憶體,用於程式的記憶體動態分配,C語言使用malloc從堆上分配記憶體,使用free釋放已分配的對應記憶體。自由儲存區不等於堆,如上所述,佈局new就可以不位於堆中。 在C++中,記憶體區分為5個區,分別是堆、棧、自由儲存區、全域性/靜態儲存區、常量儲存區;在C中,C記憶體區分為堆、棧、全域性/靜態儲存區、常量儲存區;)

3.引數:

new申請記憶體分配時無須指定記憶體塊的大小,編譯器會根據型別資訊自行計算。而malloc則需要顯式地指出所需記憶體的尺寸

3.返回型別:

new操作符記憶體分配成功時,返回的是物件型別的指標,型別嚴格與物件匹配,無須進行型別轉換,故new是符合型別安全性的操作符,失敗是丟擲異常。而malloc記憶體分配成功則是返回void * ,需要通過強制型別轉換將void*指標轉換成我們需要的型別,失敗時返回NULL。

4.構造和析構: 

new會先呼叫operator new函式,申請足夠的記憶體(通常底層使用malloc實現)。然後呼叫型別的建構函式,初始化成員變數,最後返回自定義型別指標。delete先呼叫解構函式,然後呼叫operator delete函式釋放記憶體(通常底層使用free實現;malloc/free是庫函式,只能動態的申請和釋放記憶體,無法強制要求其做自定義型別物件構造和析構工作。

5.過載:

C++允許過載new/delete操作符,特別的,佈局new的就不需要為物件分配記憶體,而是指定了一個地址作為記憶體起始區域,new在這段記憶體上為物件呼叫建構函式完成初始化工作,並返回此地址。而malloc不允許過載。

6.相互呼叫:

new的實現可以用malloc,malloc的實現不可以使用new;