日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

C++拷贝构造函数(复制构造函数)详解

發布時間:2025/4/5 c/c++ 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++拷贝构造函数(复制构造函数)详解 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

link

復制構造函數是構造函數的一種,也稱拷貝構造函數,它只有一個參數,參數類型是本類的引用。

如果類的設計者不寫復制構造函數,編譯器就會自動生成復制構造函數。大多數情況下,其作用是實現從源對象到目標對象逐個字節的復制,即使得目標對象的每個成員變量都變得和源對象相等。編譯器自動生成的復制構造函數稱為“默認復制構造函數”。

注意,默認構造函數(即無參構造函數)不一定存在,但是復制構造函數總是會存在

#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); //用復制構造函數初始化c2cout<<c2.real<<","<<c2.imag; //輸出 1,2return 0; }
第 13 行給出了初始化 c2 的參數,即 c1。只有編譯器自動生成的那個默認復制構造函數的參數才能和 c1 匹配,因此,c2 就是以 c1 為參數,調用默認復制構造函數進行初始化的。初始化的結果是 c2 成為 c1 的復制品,即 c2 和 c1 每個成員變量的值都相等。
如果編寫了復制構造函數,則默認復制構造函數就不存在了。下面是一個非默認復制構造函數的例子。
#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); //調用復制構造函數cout<<c2.real<<","<<c2.imag;return 0; } Copy Constructor called 1,2

第 9 行,復制構造函數的參數加不加 const 對本程序來說都一樣。但加上 const 是更好的做法,這樣復制構造函數才能接受常量對象作為參數,即才能以常量對象作為參數去初始化別的對象。

第 17 行,就是以 c1 為參數調用第 9 行的那個復制構造函數初始化的。該復制構造函數執行的結果是使 c2 和 c1 相等,此外還輸出Copy Constructor called。

可以想象,如果將第 10 行刪去或改成real = 2*c.real; imag = imag + 1;,那么 c2 的值就不會等于 c1 了。也就是說,自己編寫的復制構造函數并不一定要做復制的工作(如果只做復制工作,那么使用編譯器自動生成的默認復制構造函數就行了)。但從習慣上來講,復制構造函數還是應該完成類似于復制的工作為好,在此基礎上還可以根據需要做些別的操作。

構造函數不能以本類的對象作為唯一參數,以免和復制構造函數相混淆。例如,不能寫如下構造函數

Complex (Complex c) {...}

制構造函數被調用的三種情況

復制構造函數在以下三種情況下會被調用。
1) 當用一個對象去初始化同類的另一個對象時,會引發復制構造函數被調用。例如,下面的兩條語句都會引發復制構造函數的調用,用以初始化 c2。
Complex c2(c1); Complex c2 = c1;
這兩條語句是等價的。
2) 如果函數 F 的參數是類 A 的對象,那么當 F 被調用時,類 A 的復制構造函數將被調用。換句話說,作為形參的對象,是用復制構造函數初始化的,而且調用復制構造函數時的參數,就是調用函數時所給的實參。
#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 函數的形參 a 在初始化時調用了復制構造函數。

  • 如果函數的返冋值是類 A 的對象,則函數返冋時,類 A 的復制構造函數被調用。換言之,作為函數返回值的對象是用復制構造函數初始化 的,而調用復制構造函數時的實參,就是 return 語句所返回的對象。例如下面的程序:
  • #include<iostream> using namespace std; class A { public:int v;A(int n) { v = n; };A(const A & a) {v = a.v;cout << "Copy constructor called" << endl;} };A Func() {A a(4);return a; }int main() {cout << Func().v << endl;return 0; }

    程序的輸出結果是:
    Copy constructor called
    4

    第19行調用了 Func 函數,其返回值是一個對象,該對象就是用復制構造函數初始化的, 而且調用復制構造函數時,實參就是第 16 行 return 語句所返回的 a。復制構造函數在第 9 行確實完成了復制的工作,所以第 19 行 Func 函數的返回值和第 14 行的 a 相等。

    需要說明的是,有些編譯器出于程序執行效率的考慮,編譯的時候進行了優化,函數返回值對象就不用復制構造函數初始化了,這并不符合 C++ 的標準。上面的程序,用 Visual Studio 2010 編譯后的輸出結果如上所述,但是在 Dev C++ 4.9 中不會調用復制構造函數。把第 14 行的 a 變成全局變量,才會調用復制構造函數。對這一點,讀者不必深究。

    總結

    以上是生活随笔為你收集整理的C++拷贝构造函数(复制构造函数)详解的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。