1. 程式人生 > >C++筆試心得

C++筆試心得

class A
{
public:
	A() { cout << "建構函式" << endl; }
	~A() { cout << "解構函式" << endl; }
	A(const A&) { cout << "拷貝建構函式" << endl; }
};

int main()
{
	A *a=new (A);
	a->~A();
	return 0;
}

如果將類宣告為指標,那麼就要顯示的呼叫它的解構函式。

{
	A *a=new A[3];
	a->~A();
}

這樣的話,本來有3個物件被構造,但是結束的時候只析構了一個物件,就會造成記憶體的浪費。

void fun(char s[100])
{
	cout << sizeof(s) << endl;
}

相當於void fun(char *s) { cout << sizeof(s) << endl; }
所以,sizeof計算的是指標,所以佔用4個位元組。

int **a[3][5];
cout << sizeof(a) << endl;

就相當於a[3][5],所以,sizeof的值為 3 * 5 * 4 = 60

new和malloc的區別

  1. 屬性
    new/delete是C++關鍵字,需要編譯器支援。malloc/free是庫函式,需要標頭檔案<stdlib.c>支援c。

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

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

  4. 分配失敗
    new記憶體分配失敗時,會丟擲bac_alloc異常。malloc分配記憶體失敗時返回NULL。

  5. 自定義型別
    new會先呼叫operator new函式,申請足夠的記憶體(通常底層使用malloc實現)。然後呼叫型別的建構函式,初始化成員變數,最後返回自定義型別指標。delete先呼叫解構函式,然後呼叫operator delete函式釋放記憶體(通常底層使用free實現)。

    malloc/free是庫函式,只能動態的申請和釋放記憶體,無法強制要求其做自定義型別物件構造和析構工作。

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

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

PS
在C++中,記憶體區分為5個區,分別是堆、棧、自由儲存區、全域性/靜態儲存區、常量儲存區;

在C中,C記憶體區分為堆、棧、全域性/靜態儲存區、常量儲存區;