詳解C++中類的六大默認成員函數

一、類的默認成員函數

二、構造函數Date(形參列表)

構造函數主要完成初始化對象,相當於C語言階段寫的Init函數。

默認構造函數:無參的構造函數或全缺省的構造函數,二者隻能存在一個,同時存在類中,調用時會出現二義性。

1、構造函數的函數名和返回值

構造函數的函數名和類名相同且無返回值

2、構造函數的調用

對象實例化時,編譯器自動調用對應的構造函數且隻調用一次

3、構造函數的重載

構造函數可以重載(多種初始化方式)註意:雖然全缺省和無參的構造函數構成重載,但是調用時存在二義性。

class Date
{
public:
	//構造函數的重載
	Date()
	{
 
	}
	Date(int year,int month,int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	Date d1;//創建並通過無參構造函數初始化對象時,無需加括號,不然變成瞭函數聲明
	Date d2(2022,9,23);
	return 0;
}

4、系統生成的默認構造函數

如果我們在類中不寫構造函數,C++編譯器會在類中幫我們生成一個無參的構造函數。我們寫瞭構造函數,那麼系統將不會生成。

我們構造對象並調用構造函數時,初始化的數據是隨機值:

5、系統生成的默認構造函數的作用

系統生成默認構造函數對內置類型不處理,對自定義類型調用他的構造函數。

註意:下方代碼中,Date類中的內置類型將會調用Date類中的構造函數;Date類中的Time _t將會調用Time類中的構造函數。

6、可以在內置類型的成員變量的聲明中給缺省值

C++11中針對內置類型不處理初始化為隨機值的問題,打瞭補丁:內置類型成員變量在類中聲明可以給默認值,甚至可以給動態開辟的缺省值,缺點是不能判斷空間是否開辟成功。

註意這裡的默認值是缺省值,不是初始化。初始化是要等對象調用時才叫初始化。

class Date
{
private:
	int _year=1;
	int _month=2;
	int _day=3;
	int* arr = (int*)malloc(sizeof(int) * 5);
};

這個特性隻能用於解決默認構造函數初始化為隨機值的問題。這個特性不能解決對象的多種初始化方式(這也是構造函數支持重載的原因),構造函數該寫還是得自己寫。

7、初始化列表初始化

  • 1、對象的每個成員變量是在初始化列表部分進行初始化,而函數體內的行為是對成員函數賦初值。
  • 2、如果沒有在初始化列表裡顯示初始化某個成員函數,對於內置類型,有缺省值用缺省值,無缺省值初始化為隨機值;對於自定義類型將會調用它的默認構造函數,沒有找到默認構造就會報錯。
  • 3、引用、const、無默認構造函數的自定義類型必須通過初始化列表初始化。
  • 4、成員變量在類中聲明次序就是其在初始化列表中的初始化順序,與其在初始化列表中的先後次序無關。
Date(int year=1,int month=1,int day=1)//缺省值
    :_year(year)//成員變量的定義
    ,_month(month)
    ,_day(day)
{}//成員變量的賦初值

總結:盡量使用使用初始化列表進行初始化,在類中盡量提供默認構造函數(最好是全缺省的默認構造函數)

8、單參數構造(C++98)、多參數構造(C++11)

class Date
{
public:
	Date(int year=1,int month=1,int day=1)
		:_year(year)
		,_month(month)
		,_day(day)
	{}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	//單參數的構造,構造+拷貝,編譯器直接優化為構造C++98
	Date d1 = 2022;
	//臨時對象具有常性,構造+拷貝,編譯器不會優化
	const Date& d2 = 2023;
	//多參數的構造C++11
	Date d3 = { 2022,10,16 };
	return 0;
}
  • 1、在構造時,支持等號加參數的構造形式,實際上發生的是隱式類型轉換。2022由int類型隱式類型轉換為Date型的臨時對象,該臨時對象再將它的值拷貝構造給d1。
  • 2、2023會隱式類型轉換為Date類型的臨時對象,具有常屬性。d2是這個臨時對象的引用,所以需要加上const。這裡發生構造+拷貝構造,涉及臨時對象的引用,所以編譯器並不會發生優化。
  • 3、可以使用explicit關鍵字修飾構造函數,會禁止隱式類型轉換。

三、析構函數~Date()

析構函數:與構造函數功能相反,析構函數不是完成對對象本身的銷毀,局部對象銷毀工作是由編譯器完成的。而對象在銷毀時會自動調用析構函數,完成對象中資源的清理工作。資源包括動態開辟的空間,文件的關閉等。相當於C語言階段寫的destroy函數。

1、析構函數的函數名、參數和返回值

析構函數的函數名是類名前加~,無參無返回值類型。

2、析構函數的特點

一個類隻能有一個析構函數。若未顯式定義,系統會自動生成默認的析構函數。註意:析構函數不能重載

對象生命周期結束時,C++編譯系統系統自動調用析構函數

3、編譯器生成的默認析構函數

對於編譯器生成的默認析構函數,對自定義類型調用他的析構函數。對於內置類型,沒有需要處理資源。

註意:Date類中的內置類型會調用Date類中的析構函數;Date類中的自定義類型Time _t會調用Time的析構函數。

四、拷貝構造Date(const Date& d)

1、拷貝構造函數是構造函數的重載

拷貝構造函數:隻有單個形參,該形參是對本類類型對象的引用(一般常用const修飾),在用已存在的類類型對象創建新對象時由編譯器自動調用。

拷貝構造函數的函數名和構造函數相同無返回值,在參數上和構造函數構成重載。

2、拷貝構造函數的參數

拷貝構造函數的參數隻有一個並且是類類型對象引用。如果使用傳值傳參的方式進行拷貝構造,在傳值的過程中實參需要拷貝一份數據給形參,這個過程需要調用拷貝構造。形成層層傳值引發對象的拷貝的遞歸(有遞無歸)調用。

3、若未顯式定義,編譯器會生成默認的拷貝構造函數

默認的構造函數對於內置類型按照字節拷貝。對於自定義類型則調用它的拷貝構造函數。

4、拷貝構造函數的深淺拷貝

通過默認的拷貝構造函數構造的對象,按字節完成拷貝。這種拷貝被稱為淺拷貝(值拷貝)。

int main()
{
	Date d1(2022,9,24);
	Date d2(d1);
    return 0;
}

對於內置類型,使用淺拷貝即可,系統默認生成的就可以做到,所以我們不用動手寫拷貝構造函數。註意這裡有d1,d2兩個對象,當main函數生命周期結束時,這兩個對象均會發生一次析構,d2先析構,d1後析構。(後定義的先銷毀,類似棧的後進先出原則)

但是淺拷貝對於占用“資源”的成員變量時(例如成員變量中有動態開辟或fopen的資源),指針雖然復制瞭,但是所指向的內容卻沒有復制,析構時存在同一塊空間被釋放兩次的問題。需要進行深拷貝。深拷貝的拷貝構造函數必須自己手動實現。

class Stack
{
public:
	Stack(int capacity=100)//構造函數
	{
		_capacity = capacity;
		_top = 0;
		_arr = (int*)malloc(sizeof(int) * 5);
		if (_arr == nullptr)
		{
			perror("malloc fail");
			exit(-1);
		}
	}
	//Stack(const Stack& st)//淺拷貝,棧這個類不能用淺拷貝
	//{
	//	_capacity = st._capacity;
	//	_top = st._top;
	//	_arr = st._arr;
	//}
	Stack(const Stack& st)//深拷貝
	{
		_capacity = st._capacity;
		_top = st._top;
		_arr = (int*)malloc(sizeof(int) * st._top);
		if (_arr == nullptr)
		{
			perror("malloc fail");
			exit(-1);
		}
		memcpy(_arr, st._arr,sizeof(int)*st._top);
	}
	~Stack()//析構函數
	{
		_capacity = 0;
		_top = 0;
		free(_arr);
		_arr = nullptr;
	}
private:
	int* _arr;
	int _top;
	int _capacity;
};

棧這個類因為成員變量中有動態開辟的空間,所以要用深拷貝。

5、拷貝構造函數調用場景

  • 1、使用已存在的對象創建新對象
  • 2、函數參數類型為類的類型對象(傳值調用,實參拷貝給形參)
  • 3、函數返回值類型為類的良心對象(傳值返回)

五、賦值運算符重載Date& operator=(const Date& d )

1、賦值運算符重載

隻有賦值運算符是默認成員函數。

2、賦值運算符重載的註意事項

  • 1、參數是const T&,傳引用可以減少一次拷貝構造。
  • 2、返回值是*this的引用,引用返回,減少一次拷貝構造,有返回值是為瞭支持函數的鏈式訪問。
  • 3、務必檢查下是否支持自己給自己賦值。
  • 4、賦值運算符重載必須是類中的默認成員函數,不能寫在全局。
  • 5、系統默認生成的賦值運算符重載會完成值拷貝。

六、取地址操作符重載和const取地址操作符重載

Date* operator&()
{
    return this;
    //return nullptr;
}
const Date* operator&()const
{
    return this;
    //return nullptr;
}

不用自己寫,除非想讓別人通過取地址操作符獲取到特定值(自己在重載函數內部寫)或屏蔽類地址。

以上就是詳解C++中類的六大默認成員函數的詳細內容,更多關於C++類成員函數的資料請關註WalkonNet其它相關文章!

推薦閱讀: