c++構造函數的知識在各類c++教材上已有介紹,不過初學者每每不太注意觀察和總結其中各類構造函數的特色和用法,故在此我根據本身的c++編程經驗總結了一下c++中各類構造函數的特色,並附上例子,但願對初學者有所幫助。ios
1. 構造函數是幹什麼的c++
class Counter { public: // 類Counter的構造函數 // 特色:以類名做爲函數名,無返回類型 Counter() { m_value = 0; } private: // 數據成員 int m_value; }
該類對象被建立時,編譯系統對象分配內存空間,並自動調用該構造函數->由構造函數完成成員的初始化工做編程
eg: Counter c1;ide
編譯系統爲對象c1的每一個數據成員(m_value)分配內存空間,並調用構造函數Counter()自動地初始化對象c1的m_value值設置爲0函數
故:構造函數的做用:初始化對象的數據成員。this
2. 構造函數的種類spa
class Complex { private : double m_real; double m_imag; public: // 無參數構造函數 // 若是建立一個類你沒有寫任何構造函數,則系統會自動生成默認的無參構造函數,函數爲空,什麼都不作 // 只要你寫了一個下面的某一種構造函數,系統就不會再自動生成這樣一個默認的構造函數,若是但願有一個這樣的無參構造函數,則須要本身顯示地寫出來 Complex(void) { m_real = 0.0; m_imag = 0.0; } // 通常構造函數(也稱重載構造函數) // 通常構造函數能夠有各類參數形式,一個類能夠有多個通常構造函數,前提是參數的個數或者類型不一樣(基於c++的重載函數原理) // 例如:你還能夠寫一個 Complex( int num)的構造函數出來 // 建立對象時根據傳入的參數不一樣調用不一樣的構造函數 Complex(double real, double imag) { m_real = real; m_imag = imag; } // 複製構造函數(也稱爲拷貝構造函數) // 複製構造函數參數爲類對象自己的引用,用於根據一個已存在的對象複製出一個新的該類的對象,通常在函數中會將已存在對象的數據成員的值複製一份到新建立的對象中 // 若沒有顯示的寫複製構造函數,則系統會默認建立一個複製構造函數,但當類中有指針成員時,由系統默認建立該複製構造函數會存在風險,具體緣由請查詢有關 「淺拷貝」 、「深拷貝」的文章論述 Complex(const Complex & c) { // 將對象c中的數據成員值複製過來 m_real = c.m_real; m_img = c.m_img; } // 類型轉換構造函數,根據一個指定的類型的對象建立一個本類的對象 // 例如:下面將根據一個double類型的對象建立了一個Complex對象 Complex::Complex(double r) { m_real = r; m_imag = 0.0; } // 等號運算符重載 // 注意,這個相似複製構造函數,將=右邊的本類對象的值複製給等號左邊的對象,它不屬於構造函數,等號左右兩邊的對象必須已經被建立 // 若沒有顯示的寫=運算符重載,則系統也會建立一個默認的=運算符重載,只作一些基本的拷貝工做 Complex &operator=(const Complex &rhs) { // 首先檢測等號右邊的是否就是左邊的對象本,如果本對象自己,則直接返回 if ( this == &rhs ) { return *this; } // 複製等號右邊的成員到左邊的對象中 this->m_real = rhs.m_real; this->m_imag = rhs.m_imag; // 把等號左邊的對象再次傳出 // 目的是爲了支持連等 eg: a=b=c 系統首先運行 b=c // 而後運行 a= ( b=c的返回值,這裏應該是複製c值後的b對象) return *this; } };
下面使用上面定義的類對象來講明各個構造函數的用法:設計
void main() { // 調用了無參構造函數,數據成員初值被賦爲0.0 Complex c1,c2; // 調用通常構造函數,數據成員初值被賦爲指定值 Complex c3(1.0,2.5); // 也能夠使用下面的形式 Complex c3 = Complex(1.0,2.5); // 把c3的數據成員的值賦值給c1 // 因爲c1已經事先被建立,故此處不會調用任何構造函數 // 只會調用 = 號運算符重載函數 c1 = c3; // 調用類型轉換構造函數 // 系統首先調用類型轉換構造函數,將5.2建立爲一個本類的臨時對象,而後調用等號運算符重載,將該臨時對象賦值給c1 c2 = 5.2; // 調用拷貝構造函數( 有下面兩種調用方式) Complex c5(c2); Complex c4 = c2; // 注意和 = 運算符重載區分,這裏等號左邊的對象不是事先已經建立,故須要調用拷貝構造函數,參數爲c2 }
3. 思考與測驗指針
(1) 爲何函數中能夠直接訪問對象c的私有成員 ?orm
Complex(const Complex & c) { // 將對象c中的數據成員值複製過來 m_real = c.m_real; m_img = c.m_img; }
(2) 挑戰題,瞭解引用與傳值的區別
Complex test1(const Complex& c) { return c; } Complex test2(const Complex c) { return c; } Complex test3() { static Complex c(1.0,5.0); return c; } Complex& test4() { static Complex c(1.0,5.0); return c; } void main() { Complex a,b; // 下面函數執行過程當中各會調用幾回構造函數,調用的是什麼構造函數? test1(a); test2(a); b = test3(); b = test4(); test2(1.2); // 下面這條語句會出錯嗎? test1(1.2); //test1( Complex(1.2 )) 呢? }
4. 淺拷貝與深拷貝
上面提到,若是沒有自定義複製構造函數,則系統會建立默認的複製構造函數,但系統建立的默認複製構造函數只會執行「淺拷貝」,即將被拷貝對象的數據成員的值一一賦值給新建立的對象,若該類的數據成員中有指針成員,則會使得新的對象的指針所指向的地址與被拷貝對象的指針所指向的地址相同,delete該指針時則會致使兩次重複delete而出錯。下面是示例:
#include <iostream.h> #include <string.h> class Person { public : // 構造函數 Person(char * pN) { cout << "通常構造函數被調用 !\n"; m_pName = new char[strlen(pN) + 1]; //在堆中開闢一個內存塊存放pN所指的字符串 if(m_pName != NULL) { //若是m_pName不是空指針,則把形參指針pN所指的字符串複製給它 strcpy(m_pName ,pN); } } // 系統建立的默認複製構造函數,只作位模式拷貝 Person(Person & p) { //使兩個字符串指針指向同一地址位置 m_pName = p.m_pName; } ~Person( ) { delete m_pName; } private : char * m_pName; }; void main( ) { Person man("lujun"); Person woman(man); // 結果致使 man 和 woman 的指針都指向了同一個地址 // 函數結束析構時 // 同一個地址被delete兩次 } // 下面本身設計複製構造函數,實現「深拷貝」,即不讓指針指向同一地址,而是從新申請一塊內存給新的對象的指針數據成員 Person(Person & chs); { // 用運算符new爲新對象的指針數據成員分配空間 m_pName=new char[strlen(p.m_pName)+ 1]; if(m_pName) { // 複製內容 strcpy(m_pName ,chs.m_pName); } // 則新建立的對象的m_pName與原對象chs的m_pName再也不指向同一地址了 }