[轉]c++類的構造函數詳解

? ?c++構造函數的知識在各種c++教材上已有介紹,不過初學者往往不太注意觀察和總結其中各種構造函數的特點和用法,故在此我根據自己的c++編程經驗總結了一下c++中各種構造函數的特點,并附上例子,希望對初學者有所幫助。

c++類的構造函數詳解????????????????????????

一、 構造函數是干什么的

class?Counter
{

public:
?????????// 類Counter的構造函數
?????????// 特點:以類名作為函數名,無返回類型
???????? Counter()
???????? {
??????????????? m_value = 0;
???????? }
?????????
private:
??????
?????????// 數據成員
?????????int?m_value;
}


?????? 該類對象被創建時,編譯系統對象分配內存空間,并自動調用該構造函數->由構造函數完成成員的初始化工作

eg:??? Counter c1;
??????? 編譯系統為對象c1的每個數據成員(m_value)分配內存空間,并調用構造函數Counter( )自動地初始化對象c1的m_value值設置為0

故:

??????? 構造函數的作用:初始化對象的數據成員。


二、 構造函數的種類

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

????????
????????
}

三、思考與測驗

1. 仔細觀察復制構造函數

??????? Complex(const?Complex & c)
??????? {
????????????????// 將對象c中的數據成員值復制過來
??????????????? m_real = c.m_real;
??????????????? m_img = c.m_img;
??????? }????
????????
為什么函數中可以直接訪問對象c的私有成員?

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 )) 呢?
? }

四、附錄(淺拷貝與深拷貝)

?????? 上面提到,如果沒有自定義復制構造函數,則系統會創建默認的復制構造函數,但系統創建的默認復制構造函數只會執行“淺拷貝”,即將被拷貝對象的數據成員的值一一賦值給新創建的對象,若該類的數據成員中有指針成員,則會使得新的對象的指針所指向的地址與被拷貝對象的指針所指向的地址相同,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不再指向同一地址了
}

轉載于:https://www.cnblogs.com/hehexiaoxia/p/5101698.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/256125.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/256125.shtml
英文地址,請注明出處:http://en.pswp.cn/news/256125.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

結構體對齊問題

#pragma (1)typedef struct {};#pragma ()#pragma pack(1) 的意義是什么skypxl | 瀏覽 27771 次發布于2009-10-01 16:38最佳答案設置結構體的邊界對齊為1個字節&#xff0c;也就是所有數據在內存中是連續存儲的。比如你在C語言中定義下面這樣的結構體&#xff1a;struct s {cha…

【機器學習】——純Python建立BP模型

參考:https://blog.csdn.net/michael_f2008/article/details/103715699 https://developer.aliyun.com/article/614411 import pandas as pd import numpy as np import datetime from sklearn.utils import shuffle# 1.初始化參數 def initialize_parameters(n_x, n_h, n_y…

組件局域網中的無集線器、Windows XP、Windows 7、Windows 8的對等網

為什么要用對等網&#xff1f; 答&#xff1a;對等網采用分散管理的方式&#xff0c;網絡中的每臺計算機既作為客戶機又可作為服務器來工作&#xff0c;每個用戶都管理自己機器上的資源。 組建局域網中無集線器的對等網 組建局域網中Windows XP的對等網 組建局域網中Windows 7的…

datatable綁定comboBox顯示數據[C#]

實現功能&#xff1a;datatable綁定comboBox,在下拉菜單中顯示對應數據 實現方法&#xff1a;    //dataSet 轉 dataTable    1、生成datatable&#xff0c;并為combox綁定數據源&#xff1a;comboBox1.DataSource dt1;comboBox1.DisplayMember "用戶編碼"…

C++總結筆記(十)——堆區內存開辟數組和二級指針

文章目錄一、堆區開辟數組1. 數組指針與指針數組的區別2. 1維數組3. 2維數組二、二級指針一、堆區開辟數組 1. 數組指針與指針數組的區別 數組指針是指指向數組的指針&#xff0c;它的本體是一個指針, 聲明指針變量的時候一般用括號&#xff0c;因為括號的優先級高于[]&#…

阿里巴巴將赴NIPS 2017 3大事業部聯袂展示AI全技能

被譽為國際頂級的機器學習和神經網絡學術會議NIPS 2017將于12月4日-9日在美國加州長灘市舉行。在本屆會議上&#xff0c;阿里巴巴除有兩篇論文入選Workshop并進行Oral和Poster形式報告外&#xff0c;3大技術事業部將連續3天&#xff08;5日-7日&#xff09;在阿里展區舉行多場技…

【圖像處理】——Python實現圖像特征提取BP神經網絡實現圖像二分類

目錄 一、圖像特征提取 二、BP實現圖像二分類 1、輸入層、隱層、輸出層結點個數設置

mysql sql語句大全

1、新建用戶&#xff1a;>CREATE USER name IDENTIFIED BY ssapdrow; 2、更改密碼&#xff1a; >SET PASSWORD FOR namePASSWORD(fdddfd); 3、權限管理 >SHOW GRANTS FOR name; //查看name用戶權限 >GRANT SELECT ON db_name.* TO name;    //給name用戶db_…

C++總結筆記(十一)—— Lambda表達式的應用

文章目錄一、Lambda表達式是什么&#xff1f;二、程序示例1. 在STL中的使用一、Lambda表達式是什么&#xff1f; Lambda表達式時C11引入的語法&#xff0c;本質上是一個匿名函數&#xff0c;用[ ] () {}三個符號組成表達式。 格式為&#xff1a;[capture list] (params list) …

【文件處理】——字典寫入json文件或TXT文件,讀取文件中的字典TypeError: Object of type ‘ndarray‘ is not JSON serializable錯誤解決方法

目錄 一、將字典寫入json文件 二、json文件中讀取字典 三、將字典寫入TXT文件中 四、從TXT中讀取字典 五、解決字典含數組存入json文件失敗的方法 1、存入前將數組變成列表 2、擴展類方法 一、將字典寫入json文件 import jsontest_dict {version: "1.0",exp…

js一些實用例子

1.獲取焦點選中文本內容 $("#id").focus(function(){ this.select(); }); 2.表單提交方式 A.自動提交 setTimeout(function(){表單對象.submit();},2000); B.onclick事件提交(鏈接提交) <input type"submit" value"提交" οnclick"retur…

360安全衛士加速球誤關閉某個應用軟件

近期因為工作原因須要使用金山快盤。但因為之前電腦上安裝了360安全衛士&#xff0c;在使用加速球功能的時候&#xff0c;會出現&#xff0c;誤關閉金山快盤的操作&#xff0c;怎么避免呢&#xff1f; 1、打開360主界面。找到安全防護中心&#xff0c;例如以下圖&#xff1a; 2…

C++STL總結筆記(一)—— 容器和容器適配器

文章目錄前言一、概念1.1 順序容器1.2 容器適配器1.3 關聯容器二、程序示例1. vector和Set自定義數據類型的訪問2.vector容器嵌套3.list容器排序4.pair對組的使用總結前言 STL是C中的基于數據結構和算法的標準模板庫&#xff0c;可以大量節約系統開發時間&#xff0c;增加程序…

【圖像處理】——比特平面原理和實現方法(全網較全面,含所有比特位圖的分層方法)

目錄 一、比特平面 1st比特平面:括號里面的為比特值,前面的是原圖像中對應要變為0的像素值

20145238-荊玉茗 《信息安全系統設計基礎》第7周學習總結

20145238 《信息安全系統設計基礎》第7周學習總結 教材學習內容總結 存儲器系統 : 存儲器系統是一個具有不同容量、成本和訪問時間的存儲設備的層次結構。 1、RAM隨機訪問存儲器 分類&#xff1a;SRAM&#xff08;靜態&#xff09;、DRAM&#xff08;動態&#xff09;特點&…

關于兩個jar包中存在包名和類名都完全相同的jar包沖突問題

2019獨角獸企業重金招聘Python工程師標準>>> 最近弄webservice&#xff0c;我使用的jdk自帶的wsimport工具來生成客戶端代碼&#xff0c;發現生成的代碼具有編譯錯誤&#xff0c;錯誤描述如下&#xff1a; The method getPort(QName, Class<T>) in the type S…

限制文本框只能輸入數字和小數點

<input type"text" class"NumText" placeholder"只能輸入數字&#xff08;可含有小數點&#xff09;"/><style> .NumText{width:200px;} .NumText::-webkit-input-placeholder{color:#F1923C;} .NumText:-moz-placeholder{…

C++STL總結筆記(二)——仿函數(函數對象)

文章目錄一、概念總結一、概念 仿函數又稱函數對象&#xff0c;即重載了函數調用運算符&#xff08;&#xff09;的類的對象。 優勢&#xff1a; 1.仿函數對象的內部可以有自己的狀態&#xff0c;可以實現一些其他的功能。 2.函數對象可以作為參數進行傳遞。 當仿函數類內重載…

【圖像處理】——灰度變換心得(cv2.normalize規范化值0-255,cv2.convertScaleAbs(new_img)轉為8位圖)

目錄 一、灰度變換函數 對數變換 加碼變換 常見雷點 常見灰度變換函數