C++構造函數+復制構造函數+重載等號運算符調用

前言:

初學C++發現瞭下面這個問題,其中Duck是一個已知的類,並以多種方式指定對象的值:

Duck d1();
Duck d2(d1);
Duck d3 = d1;
Duck d4;
d4 = d1;

問題在於,上述d1、d2、d3、d4是如何創建的呢?分別調用的哪個函數呢?

1、賦值和初始化的區別

C++中,賦值和初始化是兩個不同的概念:

  • 初始化是指對象創建之時指定其初值,分為直接初始化和復制初始化兩種(一句);
  • 賦值是指對象創建以後指定其值(兩句);

對於上述d1、d2、d3、d4的創建來說,分別對應下述情況:

Duck d1();    // 直接初始化
Duck d2(d1);  // 復制初始化
Duck d3 = d1; // 復制初始化
Duck d4;      // 未初始化
d4 = d1;      // 賦值

2、初始化和賦值分別調用哪個函數?

直接初始化時,參數為對象所需的值,此時調用構造函數;復制初始化時,參數是一個已存在的類對象,此時調用復制構造函數;賦值時,對象是已經定義好的,調用“重載的等號賦值操作”,使用另一個對象的值計算出此對象的值。

Duck d1();    // 調用構造函數
Duck d2(d1);  // 調用復制構造函數
Duck d3 = d1; // 調用復制構造函數
Duck d4;      // 
d4 = d1;      // 賦值

需要特別註意的是下面兩種方式都是初始化,而且都是調用的復制構造函數,容易誤以為第二種方式是賦值:

Duck d2(d1);
Duck d3 = d1;

3、編寫測試類

對於上述Duck類的初始化以及賦值方式,為瞭正確的測定每種情況分別調用瞭什麼函數,下面的類中包含瞭構造函數、復制構造函數、重載等號賦值運算符函數:

class Duck{
public:
    Duck() { cout << "constructing !!!" << endl; };
    Duck(const Duck &d) { cout << "copying !!!" << endl; }
    Duck& operator=(const Duck &another);
};

Duck& Duck::operator=(const Duck &another) {
    cout << "operator= !!!" << endl;
    return *this;
}

使用Section 1Section 2中的各種方式指定對象的值時,可以方便的確定出各個函數的調用情況。

到此這篇關於C++構造函數+復制構造函數+重載等號運算符調用的文章就介紹到這瞭,更多相關C++函數調用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: