国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C++ > 正文

C++構造函數和析構函數

2019-11-06 06:10:30
字體:
來源:轉載
供稿:網友

構造函數和析構函數

一.構造函數

構造函數時一種特殊的函數,它主要用于為對象分配空間,進行初始化。

(注:構造函數沒有this指針)

1.      構造函數的幾個特點:

函數名與類名相同

參數任意,但是沒有返回值,viod也不行

它是在實例化對象的時候自動的調用,而不需要用戶調用

構造函數可以重載

構造函數可以寫在類體內,也可以寫在類體外

2.      構造函數的調用形式

(1)     類名對象名[(實參表)]

(2)     類名 *指針變量名 = new 類名[(實參表)]

下面是構造函數的使用方法

#include<iostream>usingnamespace std; classDate{PRivate:    int year;    int month;    int day;public:    Date(int y,int m,int d);    voidsetDate(int y,int m,int d);    voidshowDate();}; Date::Date(inty, intm, intd){    cout << "Constcuting..." << endl;    year = y;    month = m;    day = d;} voidDate::setDate(inty, intm, intd){    year = y;    month = m;    day = d;} inlinevoidDate::showDate(){    cout <<year << "." << month << "."<< day << endl;} int main(){    Datedate1(2017,3,2);    cout << "date1:" << endl;    date1.showDate();    date1.setDate(2017,3,3);    cout << "date2:" << endl;    date1.showDate();    return 0;}上面代碼中主函數中的構造函數的使用是采用第一種方式,這里再提供第二種方式int main(){    Date *pdate;    pdate = newDate(2017,2,3);    cout << "Date1" << endl;    pdate->showDate();    pdate->setDate(2017,2,3);    cout << "Date2" << endl;    pdate->showDate();    return 0; }

這段代碼中編譯器開辟了一個存儲空間,并且存放了一個Date類,但是這個對象沒有名字,成為無名對象,但是該對象有地址,這個地址存放在pdate中,我們通過指針可以找到他,訪問時用new動態建立的對象一般是不用對象名的,而是通過指針進行訪問。如果不需要時,可以通過delete進行釋放。

二. 成員初始化列表

  C++還提供了另一種初始化成員的方法——用成員初始化列表來實現對數據成員的初始化,這種方法不在函數體內用賦值語句,而是在函數首部實現的。

例如在構造函數的定義中可以使用如下的方式:

Date::Date(inty, intm, intd) :year(y),month(m), day(d){    cout << "Constcuting..." << endl;}

帶有成員列表的構造函數一般的形式如下:

類名 ::構造函數名([參數表]):[(成員初始化列表)]

{}

成員初始化列表的一般形式為:

數據成員名1(初始值1),數據成員名2(初始值2),…

成員初始化列表有什么用途呢,一般對于const修飾的數據成員,或者是引用類型的數據成員。

注意:使用成員初始化列表初始化的時候,它的初始化的順序是按照在類中聲明的順序進行初始化的,而不是按照成員初始化列表中的順序進行初始化。

三. 帶默認參數的構造函數(帶缺省參數)

   在構造函數中,有一些成員值是不變的,這時我們可以使用帶默認參數的構造函數

   意思就是我們在定義構造函數時,可以在形參的部分隊參數進行賦初值。

四. 析構函數

析構函數也是一種特殊的成員函數,它通常用于撤銷對象時的一些清理任務

析構函數的特點如下:

析構函數和構造函數名字相同,但是它的前面必須加一個波浪號(~)

析構函數沒有參數,也沒有返回值,而且不能重載。

析構函數自動被調用

例子:

同樣是上文的Date類,我們可以在定義類的時候定義析構函數,如

class Date

{

  …

  ~Date();

}

 

Date::~Date()

{

 cout<<”destruting …”<<endl;

}

 

在以下情況下,當對象的聲明周期結束時,析構函數會被自動調用

(1). 如果定義了一個全局對象,則在程序流離開作用域(如main()函數結束或者調用exit()函數)時,調用該全局對象的析構函數。

(2). 如果一個對象被定義在一個函數體內,則當這個函數調用結束時,該函數應該釋放,析構函數自動被調用

(3). 若一個對象是使用new運算符進行動態創建的,在使用delete運算符釋放它時,delete會調用析構函數。

五. 默認的構造函數和默認的析構函數

   1.默認的構造函數(系統自帶的構造函數,全參數構造函數,無參的構造函數都可以叫默認的構造函數)

   一般寫程序時會定義構造函數,但是如果沒有定義構造函數,系統會自動生成一個構造函數,這就是默認的構造函數。上面的程序中,如果沒有定義構造函數,而直接使用Date date1;這時系統會為Date類生成下面形式的構造函數:

Date ::Date()

{}

并且使用這個默認的構造函數對date1進行初始化,但是這個構造函數沒有任何參數,它只能開辟一個存儲空間,而不能給對象中的數據成員賦值,這時的初始值是個隨機數,程序運行的時候可能會造成錯誤。

補充說明:對沒有定義構造函數的類,其公有數據成員可以用初始值列表進行初始化。

      只要一個類定義了一個構造函數,系統將不再給它提供默認的構造函數

3.      默認的析構函數

每個類都有一個析構函數,如果一個類沒有定義析構函數,那么編譯系統會自動的生成一個析構函數。

注意:在C++調用構造函數的時候注意不能出現這種形式例如有一個類Date,這個時候實例化一個對象Date date();這里不是調用了這個構造函數,而是一個函數的聲明,是錯誤的,如果不給這個對象傳遞參數就不要寫后面的括號,傳遞參數的時候寫括號,后面在加入參數。

六.拷貝構造函數

     拷貝構造函數的形參是本類對象的引用,拷貝構造函數的作用是在建立一個新得對象時,使用一個已經存在的對象去初始化這個新對象。形如:Point p2(p1);

     拷貝構造函數的幾個特點:

1.      因為拷貝構造函數也是構造函數,所以它的函數名必須與類名相同,而且也沒有返回值

2.      拷貝構造函數只有一個參數,而且是同類對象的引用

3.      每個類都有一個拷貝構造函數,可以自己定義用于初始化新的對象,如果沒有定義系統會自動的定義,用于復制與數據成員值相同的對象。

拷貝構造函數的使用:

1.      自定義拷貝構造函數

自定義拷貝構造函數的一般形式如下:

類名::類名(const 類名 &對象名)

 

自定義拷貝構造函數的調用:

代入法:類名 對象2(對象1);

賦值發:類名 對象2 = 對象1;

2.      默認的拷貝構造函數

如果用戶沒有定義自定義的拷貝構造函數,然后又使用了拷貝構造函數,系統會調用默認的拷貝構造函數,例如Rectangle p2(p1);此時會把p1中各個域的值均復制給p2

3.      調用拷貝構造函數的三種情況

(1)    當用類的一個對象去初始化類的另一個對象的時候

(2)    當函數的形參是類的對象,調用函數進行形參和實參結合時

(3)    當函數的返回值是對象,函數執行完成返回調用者時

        下面是具體的程序,還有注釋部分,注釋部分就是上面各種的講解

 

   #include<iostream>usingnamespace std; classRectangle{private:    int_length;    int _width;public:    Rectangle(int len =10, int wid =10); //構造函數    Rectangle(constRectangle&p);  //拷貝構造函數    void disp();}; Rectangle::Rectangle(intlen, intwid)  //構造函數{    _length = len;    _width = wid;    cout << "usingnormal constructor" << endl;} Rectangle::Rectangle(constRectangle &p)  //拷貝構造函數{    _length = 2 * p._length;    _width = 2 * p._width;    cout << "usingcopy constructor" << endl;} voidRectangle::disp(){    cout <<_length << " " << _width << endl;} void fun1(Rectanglep){    p.disp();} Rectangle fun2(){    Rectanglep4(10,30);     //這里調用了普通的構造函數    returnp4;             //這里返回的一個Rectangle的對象,我們會使用p2 = fun2();去接受他的返回值,所以這里相當于給p2用了一個拷貝構造函數} int main(){    Rectangle p1(30,40);  //定義了p1,調用構造函數    p1.disp();    Rectanglep2(p1);   //調用拷貝構造函數把p1里面的值全部復制給p2(情況1)    p2.disp();    Rectangle p3 =p1;  //調用拷貝構造函數,把p1的值復制給p3(情況1)    p3.disp();    fun1(p1);   //這個時候傳入的是p1的一份引用,然后會調用Rectangle的拷貝構造函數                //這里應該思考的一個問題就是,為什么調用的是Rectangle的拷貝構造函數,這里我的一個猜測是,直接傳一個對象的時候,就像傳數組名一樣了,發生了一個轉換之類的                //可能就把對象轉換成對象的一個引用了吧    p1.disp();    p2 =fun2();   //函數的返回值是對象,屬于第三種情況調用拷貝構造函數    p2.disp();    system("pause");    return 0;}

 

六. 淺拷貝和深拷貝

 所謂淺拷貝,就是如果沒有自定義拷貝構造函數而直接使用的話,就只是單純的進行值得賦值,但是當類的數據成員有指針類型的時候,并且在使用指針的時候,給這個指針動態的開辟了一個新的循存儲空間的話,當我實例化一個對象a之后,a里面的一個數據成員p動態的開辟了一個存儲空間,然后我有實例化b,而且是通過調用未定義的拷貝構造函數,這樣就直接把a里面的內容給了b,b里面的一個指針也指向剛剛a里面的內存空間,這時候就出現了一種情況就是當我調用析構函數清理a 的內存空間的時候,因為b中的指針也指向了那片空間,所以此時b中的那個指針就沒有意義了,這就造成了錯誤。所以建議再使用指針的相關操作的時候,盡量去自定義拷貝構造函數。

 

  


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 大名县| 涪陵区| 彭泽县| 许昌市| 娄底市| 乌什县| 萨嘎县| 嵊泗县| 同心县| 大余县| 保靖县| 鲁甸县| 南投县| 康乐县| 武鸣县| 衡东县| 阿拉善右旗| 河东区| 禄劝| 丽江市| 江源县| 渭源县| 龙川县| 达尔| 昌江| 龙江县| 大余县| 雷州市| 和田市| 丰镇市| 抚远县| 临漳县| 玛多县| 金阳县| 黄梅县| 玉门市| 西乌| 武山县| 额济纳旗| 华阴市| 新津县|