c++基礎(chǔ)語(yǔ)法:構(gòu)造函數(shù)與析構(gòu)函數(shù)
一.構(gòu)造函數(shù)
類似于java,C++中也有構(gòu)造函數(shù)的概念,相關(guān)用法如下:
1.1 構(gòu)造函數(shù)的定義
#include <iostream> using namespace std; class Student{ private: char *m_name; int m_age; float m_score; public: //聲明構(gòu)造函數(shù) Student(char *name, int age, float score); //聲明普通成員函數(shù) void show(); }; //定義構(gòu)造函數(shù) Student::Student(char *name, int age, float score){ m_name = name; m_age = age; m_score = score; } //定義普通成員函數(shù) void Student::show(){ cout<<m_name<<"的年齡是"<<m_age<<",成績(jī)是"<<m_score<<endl; } int main(){ //創(chuàng)建對(duì)象時(shí)向構(gòu)造函數(shù)傳參 Student stu("小明", 15, 92.5f); stu.show(); //創(chuàng)建對(duì)象時(shí)向構(gòu)造函數(shù)傳參 Student *pstu = new Student("李華", 16, 96); pstu -> show(); return 0; }
運(yùn)行結(jié)果:
小明的年齡是15,成績(jī)是92.5
李華的年齡是16,成績(jī)是96
1.2 構(gòu)造函數(shù)的重載
構(gòu)造函數(shù)同樣也支持重載操作:
#include <iostream> using namespace std; class Student{ private: char *m_name; int m_age; float m_score; public: Student(); Student(char *name, int age, float score); void setname(char *name); void setage(int age); void setscore(float score); void show(); }; Student::Student(){ m_name = NULL; m_age = 0; m_score = 0.0; } Student::Student(char *name, int age, float score){ m_name = name; m_age = age; m_score = score; } void Student::setname(char *name){ m_name = name; } void Student::setage(int age){ m_age = age; } void Student::setscore(float score){ m_score = score; } void Student::show(){ if(m_name == NULL || m_age <= 0){ cout<<"成員變量還未初始化"<<endl; }else{ cout<<m_name<<"的年齡是"<<m_age<<",成績(jī)是"<<m_score<<endl; } } int main(){ //調(diào)用構(gòu)造函數(shù) Student(char *, int, float) Student stu("小明", 15, 92.5f); stu.show(); //調(diào)用構(gòu)造函數(shù) Student() Student *pstu = new Student(); pstu -> show(); pstu -> setname("李華"); pstu -> setage(16); pstu -> setscore(96); pstu -> show(); return 0; }
運(yùn)行結(jié)果:
小明的年齡是15,成績(jī)是92.5
成員變量還未初始化
李華的年齡是16,成績(jī)是96
1.3 默認(rèn)構(gòu)造函數(shù)
類似于java,如果用戶自己沒(méi)有定義構(gòu)造函數(shù),那么編譯器會(huì)自動(dòng)生成一個(gè)默認(rèn)的構(gòu)造函數(shù),只是這個(gè)構(gòu)造函數(shù)的函數(shù)體是空的。
注意:調(diào)用沒(méi)有參數(shù)的構(gòu)造函數(shù)也可以省略括號(hào)。
Student *stu = new Student;
Student *stu = new Student();
以上兩種寫法是等價(jià)的。
1.4 構(gòu)造函數(shù)的參數(shù)初始化表
構(gòu)造函數(shù)的主要目的是用于對(duì)成員變量進(jìn)行初始化, 為了達(dá)到這個(gè)目的,可以在構(gòu)造函數(shù)的函數(shù)體中對(duì)成員變量一一賦值,還可以采用參數(shù)初始化表。具體寫法如下:
#include <iostream> using namespace std; class Student{ private: char *m_name; int m_age; float m_score; public: Student(char *name, int age, float score); void show(); }; //采用參數(shù)初始化表 Student::Student(char *name, int age, float score): m_name(name), m_age(age), m_score(score){ //TODO: } void Student::show(){ cout<<m_name<<"的年齡是"<<m_age<<",成績(jī)是"<<m_score<<endl; } int main(){ Student stu("小明", 15, 92.5f); stu.show(); Student *pstu = new Student("李華", 16, 96); pstu -> show(); return 0; }
運(yùn)行結(jié)果:
小明的年齡是15,成績(jī)是92.5
李華的年齡是16,成績(jī)是96
1.5 使用參數(shù)初始化表來(lái)初始化const成員變量
參數(shù)初始化表還有一個(gè)很重要的作用,那就是初始化 const 成員變量。初始化 const 成員變量的唯一方法就是使用參數(shù)初始化表。例:
class VLA { private: const int m_len; int *m_arr; public: VLA(int len); }; VLA::VLA(int len):m_len(len) { m_arr = new int[len]; }
二.析構(gòu)函數(shù)
創(chuàng)建對(duì)象時(shí)系統(tǒng)會(huì)自動(dòng)調(diào)用構(gòu)造函數(shù)進(jìn)行初始化工作,同樣,銷毀對(duì)象時(shí)系統(tǒng)也會(huì)自動(dòng)調(diào)用一個(gè)函數(shù)來(lái)進(jìn)行清理工作,例如釋放分配的內(nèi)存、關(guān)閉打開的文件等,這個(gè)函數(shù)就是析構(gòu)函數(shù)。
析構(gòu)函數(shù)(Destructor)也是一種特殊的成員函數(shù),沒(méi)有返回值,不需要程序員顯式調(diào)用(程序員也沒(méi)法顯式調(diào)用),而是在銷毀對(duì)象時(shí)自動(dòng)執(zhí)行。構(gòu)造函數(shù)的名字和類名相同,而析構(gòu)函數(shù)的名字是在類名前面加一個(gè)~符號(hào)。
注意:析構(gòu)函數(shù)沒(méi)有參數(shù),不能被重載,因此一個(gè)類只能有一個(gè)析構(gòu)函數(shù)。如果用戶沒(méi)有定義,編譯器會(huì)自動(dòng)生成一個(gè)默認(rèn)的析構(gòu)函數(shù)。
class VLA { public: VLA(int len); ~VLA(); // 析構(gòu)函數(shù) private: const int m_len; int *m_arr; }; VLA::VLA(int len):m_len(len) { // 構(gòu)造函數(shù)初始化 if (len > 0) {m_arrz = new int[len];}; else {m_arr = NULL;}; } VLA::~VLA() { delete []m_arr; // 在析構(gòu)函數(shù)中釋放堆區(qū)申請(qǐng)的內(nèi)存 }
C++ 中的 new 和 delete 分別用來(lái)分配和釋放內(nèi)存,它們與C語(yǔ)言中 malloc()、free() 最大的一個(gè)不同之處在于:用 new 分配內(nèi)存時(shí)會(huì)調(diào)用構(gòu)造函數(shù),用 delete 釋放內(nèi)存時(shí)會(huì)調(diào)用析構(gòu)函數(shù)。構(gòu)造函數(shù)和析構(gòu)函數(shù)對(duì)于類來(lái)說(shuō)是不可或缺的,所以在C++中我們非常鼓勵(lì)使用 new 和 delete。
下面是其他網(wǎng)友的補(bǔ)充
說(shuō)實(shí)話c++還是以前在學(xué)校的時(shí)候用過(guò)的,從畢業(yè)到現(xiàn)在一直用c嵌入式編程,現(xiàn)在重新搬出C++語(yǔ)法 ,如果理解上有錯(cuò)誤的地方,還請(qǐng)路過(guò)的朋友多指正~~~
構(gòu)造函數(shù)用來(lái)構(gòu)造一個(gè)對(duì)象,主要完成一些初始化工作,如果類中不提供構(gòu)造函數(shù),編譯器會(huì)默認(rèn)的提供一個(gè)默認(rèn)構(gòu)造函數(shù)(參數(shù)為空的構(gòu)造函數(shù)就是默認(rèn)構(gòu)造函數(shù)) ;析構(gòu)函數(shù)是隱式調(diào)用的,delete對(duì)象時(shí)候會(huì)自動(dòng)調(diào)用完成對(duì)象的清理工作。
現(xiàn)在主要看看繼承中的構(gòu)造函數(shù)和析構(gòu)函數(shù)的調(diào)用:
class A {} ; class B : public A {}; class C : public B {}; c * ptr = new C() ; delete ptr ;
一般來(lái)說(shuō),上面的代碼構(gòu)造函數(shù)是先調(diào)用最根父類的構(gòu)造函數(shù),然后是次一級(jí)父類構(gòu)造函數(shù),依次而來(lái)直到派生類本身的構(gòu)造函數(shù),而且對(duì)父類構(gòu)造函數(shù)的調(diào)用都是父類的默認(rèn)構(gòu)造函數(shù)(當(dāng)然也可以顯示地調(diào)用父類的非默認(rèn)構(gòu)造函數(shù)),也就是說(shuō)派生類在構(gòu)造本身之前會(huì)首先把繼承來(lái)的父類成分先構(gòu)造好;
對(duì)析構(gòu)函數(shù)的調(diào)用是先調(diào)用派生類本身的析構(gòu)函數(shù),然后是上一層父類析構(gòu)函數(shù),直到根父類析構(gòu)函數(shù) ,當(dāng)沒(méi)有多態(tài)的時(shí)候,析構(gòu)函數(shù)是這樣調(diào)用的。
改一下上面的代碼:
A * ptr = new C() ;
delete ptr ;
在多態(tài)的情況下,如果基類A中的析構(gòu)函數(shù)不是虛構(gòu)造函數(shù),則當(dāng)delete ptr的時(shí)候只會(huì)調(diào)用A的析構(gòu)函數(shù),不會(huì)調(diào)用B和C中的析構(gòu)函數(shù);如果A中的析構(gòu)函數(shù)是虛構(gòu)造函數(shù)就會(huì)調(diào)用所有的析構(gòu)函數(shù),調(diào)用順序和一般情況一樣。
再改一下上面的代碼:
B *prt = new C();
delete ptr ;
在多態(tài)的情況下,如果A,B中的析構(gòu)函數(shù)都不是虛析構(gòu)函數(shù),則當(dāng)delete ptr的時(shí)候先調(diào)用B的析構(gòu)函數(shù),再調(diào)A的析構(gòu)函數(shù),不會(huì)調(diào)用C中的析構(gòu)函數(shù),如果A或者B中至少有一個(gè)是虛析構(gòu)函數(shù),則析構(gòu)函數(shù)調(diào)用和一般情況一樣。
因此總結(jié)一下規(guī)律:
CA * ptr = new CB() ;
delete ptr ;
CB是CA的子類,構(gòu)造函數(shù)的調(diào)用一直是一樣的,當(dāng)具備多態(tài)的時(shí)候:
如果CA及其父類都不具備虛析構(gòu)函數(shù),則首先調(diào)用A的析構(gòu)函數(shù),然后調(diào)用A的父類析構(gòu)函數(shù)直到根父類析構(gòu)函數(shù),不會(huì)調(diào)用A以下直到派生類的析構(gòu)函數(shù) ;如果如果CA及其父類只要有一個(gè)具備虛析構(gòu)函數(shù),則析構(gòu)函數(shù)調(diào)用跟一般情況一樣。
因此:帶有多態(tài)性質(zhì)的基類應(yīng)該聲明虛析構(gòu)函數(shù) ,這樣的基類一般還有其他虛函數(shù);
如果類的設(shè)計(jì)不是用于基類,而且不具備多態(tài)性,則析構(gòu)函數(shù)不應(yīng)該聲明為虛析構(gòu)函數(shù)
小測(cè)試代碼:
#include<iostream> using namespace std ; class A { public: A(){cout<<"A constructor"<<endl;} A(char * arp) { cout <<"not default " ;} ~CA(){cout<<"A desstructor"<<endl;} }; class B:public A { public: B(){cout<<"B constructor"<<endl;} ~B(){cout<<"B desstructor"<<endl;} }; class C:public B { public: C(char * arp){cout<<"C constructor"<<endl;} ~C(){cout<<"C desstructor"<<endl;} }; int main() { C * ptr = new C("hello world") ; delete ptr ; }
另外effective C++中提到的:
1、析構(gòu)函數(shù)不能吐出異常,如果析構(gòu)函數(shù)掉用的函數(shù)可能產(chǎn)生異常,要在析構(gòu)函數(shù)內(nèi)部進(jìn)行捕獲進(jìn)行處理,因?yàn)槿绻鰳?gòu)函數(shù)拋出異常的話,比如說(shuō)vector,當(dāng)調(diào)用各個(gè)對(duì)象的析構(gòu)函數(shù)進(jìn)行刪除的時(shí)候可能導(dǎo)致拋出多個(gè)異常,從而使程序進(jìn)入不確定狀態(tài)。
2、如果用戶需要對(duì)某個(gè)操作函數(shù)運(yùn)行期間拋出的異常作出反應(yīng),那么class應(yīng)該提供一個(gè)普通函數(shù)執(zhí)行這個(gè)操作。
3、在構(gòu)造函數(shù)和析構(gòu)函數(shù)中都不應(yīng)該調(diào)用虛函數(shù),這是因?yàn)楫?dāng)調(diào)用構(gòu)造函數(shù)構(gòu)造對(duì)象的時(shí)候,首先會(huì)調(diào)用父類的構(gòu)造函數(shù),此時(shí)對(duì)象的類型在編譯器看來(lái)就是一個(gè)父類對(duì)象(實(shí)際此時(shí)子類成員還處于不確定狀態(tài)),會(huì)調(diào)用父類的虛函數(shù),而不會(huì)調(diào)用子類的虛函數(shù)。
- C++編程析構(gòu)函數(shù)拷貝構(gòu)造函數(shù)使用示例詳解
- C++中構(gòu)造函數(shù)與析構(gòu)函數(shù)的詳解及其作用介紹
- 正確理解C++的構(gòu)造函數(shù)和析構(gòu)函數(shù)
- C++語(yǔ)法詳解之封裝、構(gòu)造函數(shù)、析構(gòu)函數(shù)
- C++構(gòu)造函數(shù)和析構(gòu)函數(shù)的使用與講解
- C++中構(gòu)造函數(shù)與析構(gòu)函數(shù)的調(diào)用順序詳解
- C++類成員構(gòu)造函數(shù)和析構(gòu)函數(shù)順序示例詳細(xì)講解
- 深入解析C++中的構(gòu)造函數(shù)和析構(gòu)函數(shù)
- C++超詳細(xì)講解構(gòu)造函數(shù)與析構(gòu)函數(shù)的用法及實(shí)現(xiàn)
相關(guān)文章
詳解C++語(yǔ)言中的加法運(yùn)算符與賦值運(yùn)算符的用法
這篇文章主要介紹了C++語(yǔ)言中的加法運(yùn)算符與賦值運(yùn)算符的用法,是C++入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2016-01-01C++?解決求兩個(gè)鏈表的第一個(gè)公共結(jié)點(diǎn)問(wèn)題
本文主要介紹了利用C++實(shí)現(xiàn)輸入兩個(gè)無(wú)環(huán)的單向鏈表時(shí),找出它們的第一個(gè)公共結(jié)點(diǎn)的問(wèn)題。文章中的示例代碼簡(jiǎn)潔易懂,感興趣的同學(xué)可以和小編一起學(xué)習(xí)一下2021-12-12C++實(shí)現(xiàn)LeetCode(45.跳躍游戲之二)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(45.跳躍游戲之二),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07C++ 中dynamic_cast<>的使用方法小結(jié)
將一個(gè)基類對(duì)象指針(或引用)cast到繼承類指針,dynamic_cast會(huì)根據(jù)基類指針是否真正指向繼承類指針來(lái)做相應(yīng)處理2013-03-03C++?ASIO實(shí)現(xiàn)異步套接字管理詳解
Boost?ASIO(Asynchronous?I/O)是一個(gè)用于異步I/O操作的C++庫(kù),該框架提供了一種方便的方式來(lái)處理網(wǎng)絡(luò)通信、多線程編程和異步操作,本文介紹了如何通過(guò)ASIO框架實(shí)現(xiàn)一個(gè)簡(jiǎn)單的異步網(wǎng)絡(luò)套接字應(yīng)用程序,需要的可以參考下2023-08-08Unix下C程序內(nèi)存泄漏檢測(cè)工具Valgrind的安裝與使用詳解
以下是對(duì)Unix下C程序內(nèi)存泄漏檢測(cè)工具Valgrind的安裝與使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過(guò)來(lái)參考下2013-08-08C語(yǔ)言編程之預(yù)處理過(guò)程與define及條件編譯
這篇文章主要為大家介紹了C語(yǔ)言編程之預(yù)處理過(guò)程與define及條件編譯,文中通過(guò)圖文及示例代碼方式作了詳細(xì)的解釋,有需要的朋友可以借鑒參考下2021-09-09