溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++語法詳解之封裝、構(gòu)造函數(shù)、析構(gòu)函數(shù)

發(fā)布時(shí)間:2020-09-30 02:29:54 來源:腳本之家 閱讀:128 作者:mrr 欄目:編程語言

大家先了解下什么是構(gòu)造函數(shù),什么是析構(gòu)函數(shù),作用是什么?

構(gòu)造函數(shù)(方法)是對象創(chuàng)建完成后第一個(gè)被對象自動(dòng)調(diào)用的方法。它存在于每個(gè)聲明的類中,是一個(gè)特殊的成員方法。作用是執(zhí)行一些初始化的任務(wù)。Php中使用__construct()聲明構(gòu)造方法,并且只能聲明一個(gè)。

析構(gòu)函數(shù)(方法)作用和構(gòu)造方法正好相反,是對象被銷毀之前最后一個(gè)被對象自動(dòng)調(diào)用的方法。是PHP5中新添加的內(nèi)容作用是用于實(shí)現(xiàn)在銷毀一個(gè)對象之前執(zhí)行一些特定的操作,諸如關(guān)閉文件和釋放內(nèi)存等。

下面在通過具體例子看下C++語法詳解之封裝、構(gòu)造函數(shù)、析構(gòu)函數(shù)。

成員變量私有化,提供公共的getter和setter給外界去訪問成員變量

class Person {
 int age;
 
public:
 void setAge(int age){
  this->age = age;
 }
 
 int getAge(){
  return this->age;
 }
 
};

int main(){
 Person person;
 person.setAge(10);
 cout << person.getAge() << endl;
}

堆空間

在程序運(yùn)行過程,為了能夠自由控制內(nèi)存的生命周期、大小,會經(jīng)常使用堆空間的內(nèi)存

堆空間的申請\釋放

malloc \ free
new \ delete
new [] \ delete []

注意

  • 申請堆空間成功后,會返回那一段內(nèi)存空間的地址
  • 申請和釋放必須是1對1的關(guān)系,不然可能會存在內(nèi)存泄露

現(xiàn)在的很多高級編程語言不需要開發(fā)人員去管理內(nèi)存(比如Java),屏蔽了很多內(nèi)存細(xì)節(jié),利弊同時(shí)存在

  • 利:提高開發(fā)效率,避免內(nèi)存使用不當(dāng)或泄露
  • 弊:不利于開發(fā)人員了解本質(zhì),永遠(yuǎn)停留在API調(diào)用和表層語法糖,對性能優(yōu)化無從下手

例如開盤int類型的空間,使用完之后銷毀

int *p = (int *)malloc(sizeof(int));
 *p = 10;
 free(p);
 
 int *p2 = new int;
 *p2 = 20;
 delete p2;
 
 int *p3 = new int[3];
 *p = 10;
 *(p+1) = 20;
 *(p+2) = 30;
 delete [] (p3);

堆空間的初始化

memset

memset 函數(shù)是將較大的數(shù)據(jù)結(jié)構(gòu)(比如對象、數(shù)組等)內(nèi)存清零的比較快的方法

如下所示

 Person person;
 person.age = 10;
 person.height = 199;
 //從person的地址開始,每個(gè)字節(jié)都賦值為0
memset(&person, 0, sizeof(person));

初始化

int *p1 = (int *)malloc(sizeof(int)); //*p1 未初始化
int *p2 = (int *)malloc(sizeof(int));
memset(p2, 0, sizeof(int));//將 *p2 的每一個(gè)字節(jié)都初始化為0

如下幾種方式

int *p1 = new int;   //未初始化
int *p2 = new int();   //被初始化為0
int *p3 = new int(5);  //被初始化為5
int *p4 = new int[3];  //數(shù)組元素未被初始化
int *p5 = new int[3]();  //3個(gè)數(shù)組元素都被初始化0
int *p6 = new int[3]{};  //3個(gè)數(shù)組元素都被初始化0
int *p7 = new int[3]{5};  //數(shù)組首元素被初始化為5,其他元素被初始化為0

構(gòu)造函數(shù)(Constructor)

構(gòu)造函數(shù)(也叫構(gòu)造器),在對象創(chuàng)建的時(shí)候自動(dòng)調(diào)用,一般用于完成對象的初始化工作

特點(diǎn)

  • 函數(shù)名與類同名,無返回值(void都不能寫),可以有參數(shù),可以重載,可以有多個(gè)構(gòu)造函數(shù)
  • 一旦自定義了構(gòu)造函數(shù),必須用其中一個(gè)自定義的構(gòu)造函數(shù)來初始化對象

注意

通過malloc分配的對象不會調(diào)用構(gòu)造函數(shù)
一個(gè)廣為流傳的、很多教程\書籍都推崇的錯(cuò)誤結(jié)論:
默認(rèn)情況下,編譯器會為每一個(gè)類生成空的無參的構(gòu)造函數(shù)
正確理解:在某些特定的情況下,編譯器才會為類生成空的無參的構(gòu)造函數(shù)
比如我們自己寫2個(gè)構(gòu)造函數(shù)

class Person{
public:
 int age;
 
 Person(){
  cout << "Person()" << endl;
 }
 
 Person(int age){
   cout << "Person(int age))" << endl;
 }
};

在不同的空間調(diào)用的時(shí)候,如下區(qū)別

// 全局區(qū)
Person p1;  //調(diào)用Person()
Person p2(); //這是一個(gè)函數(shù),函數(shù)名是p2,返回值類型是Person,無參
Person p3(18); //調(diào)用 Person(int)

int main(){
 //??臻g
 Person p4;  //調(diào)用Person()
 Person p5(); //這是一個(gè)函數(shù),函數(shù)名是p5,返回值類型是Person,無參
 Person p6(18); //調(diào)用 Person(int)
 
 
 //堆空間
 Person *p7 = new Person;  //調(diào)用Person()
 Person *p8 = new Person(); //調(diào)用Person()
 Person *p9 = new Person(20); //調(diào)用 Person(int)
}

析構(gòu)函數(shù)

析構(gòu)函數(shù)(也叫析構(gòu)器),在對象銷毀的時(shí)候自動(dòng)調(diào)用,一般用于完成對象的清理工作

特點(diǎn)

函數(shù)名以~開頭,與類同名,無返回值(void都不能寫),無參,不可以重載,有且只有一個(gè)析構(gòu)函數(shù)

注意

  • 通過malloc分配的對象free的時(shí)候不會調(diào)用析構(gòu)函數(shù)
  • 構(gòu)造函數(shù)、析構(gòu)函數(shù)要聲明為public,才能被外界正常使用

例如下面的代碼

class Cat{
public:
 int age;
 Cat(){
  cout << "Cat()" << endl;
 }
 
 ~Cat(){
  cout << "~Cat()" << endl;
 }
};


class Person{
public:
 int age;
 Cat *cat;
 Person(){
  this->cat = new Cat();
  cout << "Person()" << endl;
 }
 
 ~Person(){
  cout << "~Person()" << endl;
 }
};

int main(){
 {
  Person person;
 }
 return 0;
}

輸出

Cat()
Person()
~Person()

當(dāng)person銷毀的時(shí)候,其持有的cat并沒有銷毀。

原因

當(dāng)person銷毀的時(shí)候,其指向cat對象的指針銷毀了,但是堆空間的cat對象依然存在,就會有內(nèi)存泄露。所以需要在析構(gòu)函數(shù)里面來釋放掉。類似的析構(gòu)函數(shù)在許多其他語言底層也是應(yīng)用廣泛,例如Objective-C的源碼中,大量使用析構(gòu)函數(shù)。

代碼改成如下所示:

~Person(){
  delete cat;
  cout << "~Person()" << endl;
 }

輸出

Cat()
Person()
~Cat()
~Person()

可知,cat對象才真正銷毀。

總結(jié)

到此這篇關(guān)于C++語法詳解之封裝、構(gòu)造函數(shù)、析構(gòu)函數(shù)的文章就介紹到這了,更多相關(guān)c++ 封裝構(gòu)造函數(shù)析構(gòu)函數(shù)內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI