C++拷贝构造函数(复制构造函数)详解
link
復制構造函數(shù)是構造函數(shù)的一種,也稱拷貝構造函數(shù),它只有一個參數(shù),參數(shù)類型是本類的引用。
如果類的設計者不寫復制構造函數(shù),編譯器就會自動生成復制構造函數(shù)。大多數(shù)情況下,其作用是實現(xiàn)從源對象到目標對象逐個字節(jié)的復制,即使得目標對象的每個成員變量都變得和源對象相等。編譯器自動生成的復制構造函數(shù)稱為“默認復制構造函數(shù)”。
注意,默認構造函數(shù)(即無參構造函數(shù))不一定存在,但是復制構造函數(shù)總是會存在
#include<iostream > using namespace std; class Complex { public:double real, imag;Complex(double r, double i) {real= r; imag = i;} }; int main(){Complex cl(1, 2);Complex c2 (cl); //用復制構造函數(shù)初始化c2cout<<c2.real<<","<<c2.imag; //輸出 1,2return 0; }第 13 行給出了初始化 c2 的參數(shù),即 c1。只有編譯器自動生成的那個默認復制構造函數(shù)的參數(shù)才能和 c1 匹配,因此,c2 就是以 c1 為參數(shù),調(diào)用默認復制構造函數(shù)進行初始化的。初始化的結果是 c2 成為 c1 的復制品,即 c2 和 c1 每個成員變量的值都相等。
如果編寫了復制構造函數(shù),則默認復制構造函數(shù)就不存在了。下面是一個非默認復制構造函數(shù)的例子。
#include<iostream> using namespace std; class Complex{ public:double real, imag;Complex(double r,double i){real = r; imag = i;}Complex(const Complex & c){real = c.real; imag = c.imag;cout<<"Copy Constructor called"<<endl ;} };int main(){Complex cl(1, 2);Complex c2 (cl); //調(diào)用復制構造函數(shù)cout<<c2.real<<","<<c2.imag;return 0; } Copy Constructor called 1,2第 9 行,復制構造函數(shù)的參數(shù)加不加 const 對本程序來說都一樣。但加上 const 是更好的做法,這樣復制構造函數(shù)才能接受常量對象作為參數(shù),即才能以常量對象作為參數(shù)去初始化別的對象。
第 17 行,就是以 c1 為參數(shù)調(diào)用第 9 行的那個復制構造函數(shù)初始化的。該復制構造函數(shù)執(zhí)行的結果是使 c2 和 c1 相等,此外還輸出Copy Constructor called。
可以想象,如果將第 10 行刪去或改成real = 2*c.real; imag = imag + 1;,那么 c2 的值就不會等于 c1 了。也就是說,自己編寫的復制構造函數(shù)并不一定要做復制的工作(如果只做復制工作,那么使用編譯器自動生成的默認復制構造函數(shù)就行了)。但從習慣上來講,復制構造函數(shù)還是應該完成類似于復制的工作為好,在此基礎上還可以根據(jù)需要做些別的操作。
構造函數(shù)不能以本類的對象作為唯一參數(shù),以免和復制構造函數(shù)相混淆。例如,不能寫如下構造函數(shù)
Complex (Complex c) {...}制構造函數(shù)被調(diào)用的三種情況
復制構造函數(shù)在以下三種情況下會被調(diào)用。
1) 當用一個對象去初始化同類的另一個對象時,會引發(fā)復制構造函數(shù)被調(diào)用。例如,下面的兩條語句都會引發(fā)復制構造函數(shù)的調(diào)用,用以初始化 c2。
Complex c2(c1); Complex c2 = c1;這兩條語句是等價的。
2) 如果函數(shù) F 的參數(shù)是類 A 的對象,那么當 F 被調(diào)用時,類 A 的復制構造函數(shù)將被調(diào)用。換句話說,作為形參的對象,是用復制構造函數(shù)初始化的,而且調(diào)用復制構造函數(shù)時的參數(shù),就是調(diào)用函數(shù)時所給的實參。
#include<iostream> using namespace std; class A{ public:A(){};A(A & a){cout<<"Copy constructor called"<<endl;} };void Func(A a){ }int main(){A a;Func(a);return 0; }程序的輸出結果為:
 Copy constructor called
這是因為 Func 函數(shù)的形參 a 在初始化時調(diào)用了復制構造函數(shù)。
程序的輸出結果是:
 Copy constructor called
 4
第19行調(diào)用了 Func 函數(shù),其返回值是一個對象,該對象就是用復制構造函數(shù)初始化的, 而且調(diào)用復制構造函數(shù)時,實參就是第 16 行 return 語句所返回的 a。復制構造函數(shù)在第 9 行確實完成了復制的工作,所以第 19 行 Func 函數(shù)的返回值和第 14 行的 a 相等。
需要說明的是,有些編譯器出于程序執(zhí)行效率的考慮,編譯的時候進行了優(yōu)化,函數(shù)返回值對象就不用復制構造函數(shù)初始化了,這并不符合 C++ 的標準。上面的程序,用 Visual Studio 2010 編譯后的輸出結果如上所述,但是在 Dev C++ 4.9 中不會調(diào)用復制構造函數(shù)。把第 14 行的 a 變成全局變量,才會調(diào)用復制構造函數(shù)。對這一點,讀者不必深究。
總結
以上是生活随笔為你收集整理的C++拷贝构造函数(复制构造函数)详解的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
 
                            
                        - 上一篇: c++ 析构函数 ~
- 下一篇: C++ this指针详解(精辟)
