欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++繼承模式詳解

 更新時間:2021年12月06日 09:33:01   作者:_End丶斷弦  
繼承機(jī)制是面向?qū)ο蟪绦蛟O(shè)計使代碼可以復(fù)用的最重要的手段,它允許程序員在保持原有的特性基礎(chǔ)上進(jìn)行擴(kuò)展,增加功能,這樣產(chǎn)生新的類,稱作是派生類。繼承呈現(xiàn)了面向?qū)ο蟪绦蛟O(shè)計的層析結(jié)構(gòu),體現(xiàn)了由簡單到復(fù)雜的認(rèn)知過程。繼承是類設(shè)計層次的復(fù)用。

繼承

繼承的概念

繼承機(jī)制是面向?qū)ο蟪绦蛟O(shè)計使代碼可以復(fù)用的最重要的手段,它允許程序員在保持原有類特性的基礎(chǔ)上進(jìn)行擴(kuò)展,增加功能。繼承呈現(xiàn)了面向?qū)ο蟪绦蛟O(shè)計的層次結(jié)構(gòu),體現(xiàn)了由簡單到復(fù)雜的認(rèn)知過程。

繼承的定義

上面的基類也可以叫父類,派生類也可以叫子類。

繼承關(guān)系和訪限定符

繼承方式

接下來用代碼測試上面的繼承方式

class Person
{
public :
 void Print ()
 {
 cout<<_name <<endl;
 }
protected :
 string _name = "張三" ; // 姓名
private :
 int _age = 18 ; // 年齡
};

class Student : public Person
{
protected :
 int _stunum = 22; // 學(xué)號
};

public繼承

上面是給的缺省值來測試沒寫構(gòu)造函數(shù)

s就繼承了Person的name,age,基類中private的age在物理上繼承了但在語法上但是不能訪問的。

也可以調(diào)用基類的成員函數(shù),但是不能直接訪問基類中private的成員,prootected可以在派生類中訪問,不能再在類外訪問

protected繼承

protected繼承,在類外連基類的public成員函數(shù)都不能用了,只能在派生類的類里面使用。

同樣基類中私有的不能訪問

private繼承就都是私有的了。

總結(jié):

  • 1.基類private成員在派生類中無論以什么方式繼承都是不可見的。這里的不可見是指基類的私有成員還是被繼承到了派生類對象中,但是語法上限制派生類對象不管在類里面還是類外面都不能去訪問它。
  • 2.基類private成員在派生類中是不能被訪問,如果基類成員不想在類外直接被訪問,但需要在派生類中能訪問,就定義為protected??梢钥闯霰Wo(hù)成員限定符是因繼承才出現(xiàn)的.
  • 3.基類的私有成員在子類中都是不可見的,其他成員在子類中等于權(quán)限最小的那個
  • 4.class的默認(rèn)繼承方式是private,struct默認(rèn)的繼承方式是public,最好顯示的寫出繼承方式
  • 5.在實際應(yīng)用一般使用public繼承,很少使用protected和private。

父類和子類對象賦值轉(zhuǎn)化

class Person
{
protected:
	string _name; // 姓名
	string _sex; // 性別
	int _age; // 年齡
};
class Student : public Person
{
public:
	int _No; // 學(xué)號
};

子類可以給父類,父類不能給子類,不僅可以是子類的對象,也可以是指針和引用


	Student s;
	Person p;
	p = s;
	Person *ptr = &s;//子類賦給父類指針
	Person &ref = s;//子類賦給父類引用

派生類對象 可以賦值給 基類的對象 / 基類的指針 / 基類的引用。這里有個形象的說法叫切片或者切割。寓意把派生類中父類那部分切來賦值過去。

基類對象不能賦值給派生類對象

基類的指針可以通過強(qiáng)制類型轉(zhuǎn)換賦值給派生類的指針。但是必須是基類的指針是指向派生類對象時才是安全的。等到子類中的默認(rèn)函數(shù)就會用到切片

繼承中的作用域

class Person
{
protected:
	string _name = "法外狂徒"; // 姓名
	int _num = 11; // 身份證號
};
class Student : public Person
{
public:
	void Print()
	{
		cout << " 姓名:" << _name << endl;
		cout << " 身份證號:" << _num << endl;
		cout << " 學(xué)號:" << _num << endl;
	}
protected:
	int _num = 2; // 學(xué)號
};

還有成員函數(shù)的隱藏

class A {
public:
	void fun(double x)
	{
		cout << "fun()->x"<< x << endl;
	}
};
class B : public A {
public:
	void fun(int i)
	{
		cout << "fun()->" << i << endl;
	}
};

int main()
{
	B b;
	b.fun(10);
	b.A::fun(11.1);//加作用域
	return 0;
}

父類和子類函數(shù)名相同不是重載而是隱藏,函數(shù)重載是在同一作用域,不同的作用域是隱藏

在子類成員函數(shù)中,可以使用 基類::基類成員 顯示訪問

在寫代碼中最好不要定義同名的成員

子類的默認(rèn)成員函數(shù)

在類和對象的時候講了6個默認(rèn)的成員函數(shù),現(xiàn)在子類中講4個,構(gòu)造,拷貝構(gòu)造,賦值和析構(gòu)

class Person //父類
{
public:
	Person(const char* name = "李四")
		: _name(name)
	{
		cout << "Person()" << endl;
	}

	Person(const Person& p)
		: _name(p._name)
	{
		cout << "Person(const Person& p)" << endl;
	}

	Person& operator=(const Person& p)
	{
		cout << "Person operator=(const Person& p)" << endl;
		if (this != &p)
			_name = p._name;

		return *this;
	}

	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
//子類
class Student : public Person
{
public:
	//構(gòu)造函數(shù)
	Student(const char* name, int num)
		: Person(name)//調(diào)用父類的構(gòu)造函數(shù)初始化父類的成員
		, _num(num)//初始化子類的成員
	{
		cout << "Student()" << endl;
	}
	//拷貝構(gòu)造
	Student(const Student& s)
		: Person(s)//這里就用到了切片,切父類的成員類拷貝
		, _num(s._num)//拷貝子類的
	{
		cout << "Student(const Student& s)" << endl;
	}

	Student& operator = (const Student& s)
	{
		cout << "Student& operator= (const Student& s)" << endl;
		if (this != &s)
		{
			Person::operator =(s);//調(diào)用父類的賦值
			_num = s._num;//賦值子類自己的
		}
		return *this;
	}

	~Student()
	{
		//子類的析構(gòu)函數(shù)完成清理后會自動調(diào)用父類的析構(gòu)函數(shù)
		cout << "~Student()" << endl;
	}
protected:
	int _num; //學(xué)號
};

總結(jié):

  • 派生類的構(gòu)造函數(shù)必須調(diào)用基類的構(gòu)造函數(shù)初始化基類的那一部分成員。如果基類沒有默認(rèn)的構(gòu)造函數(shù),則必須在派生類構(gòu)造函數(shù)的初始化列表階段顯示調(diào)用。
  • 派生類的拷貝構(gòu)造函數(shù)必須調(diào)用基類的拷貝構(gòu)造完成基類的拷貝初始化。
  • 派生類的operator=必須要調(diào)用基類的operator=完成基類的復(fù)制。
  • 派生類的析構(gòu)函數(shù)會在被調(diào)用完成后自動調(diào)用基類的析構(gòu)函數(shù)清理基類成員。因為這樣才能保證派生類對象先清理派生類成員再清理基類成員的順序。
  • 派生類對象初始化先調(diào)用基類構(gòu)造再調(diào)派生類構(gòu)造。
  • 派生類對象析構(gòu)清理先調(diào)用派生類析構(gòu)再調(diào)基類的析構(gòu)。

繼承與友元

友元關(guān)系不能繼承,父類友元不能訪問子類私有和保護(hù)成員

class Student;
class Person
{
public:
	friend void Display(const Person& p, const Student& s);
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	friend void Display(const Person& p, const Student& s);
protected:
	int _stuNum; // 學(xué)號
};
void Display(const Person& p, const Student& s) {
	cout << p._name << endl;//可以訪問
	cout << s._stuNum << endl;//要在子類中加上友元才能訪問,不加會報錯
}
int main()
{
	Person p;
	Student s;
	Display(p, s);
	return 0;
}

繼承與靜態(tài)成員

基類定義了static靜態(tài)成員,則整個繼承體系里面只有一個這樣的成員。無論派生出多少個子類,都只有一個static成員實例 。

class Person
{
public:
	Person() { ++_count; }
protected:
	string _name; // 姓名
public:
	static int _count; // 統(tǒng)計人的個數(shù)。
};
int Person::_count = 0;
class Student : public Person
{
protected:
	int _id; // 學(xué)號
};
class Graduate : public Student
{
protected:
	string _Course; // 科目
};

int main()
{
	Student s1;
	Student s2;
	Student s3;
	Graduate s4;
	cout << " 人數(shù) :" << Person::_count << endl;
	cout << " 人數(shù) :" << Student::_count << endl;
	cout << " 人數(shù) :" << &Person::_count << endl;
	cout << " 人數(shù) :" << &Student::_count << endl;
	return 0;
}

再加上count的地址可以看出是同一個的count。計算出子類實例化了多少個對象就可以在父類中定義個count自加。

復(fù)雜的菱形繼承

單繼承:一個子類只有一個直接父類時稱這個繼承關(guān)系為單繼承

多繼承:一個子類有兩個或以上直接父類時稱這個繼承關(guān)系為多繼承

菱形繼承:菱形繼承是多繼承的一種特殊情況。

菱形繼承的問題:從下面的對象成員模型構(gòu)造,可以看出菱形繼承有數(shù)據(jù)冗余和二義性的問題。在Assistant的對象中Person成員會有兩份。

class Person
{
public:
	string _name; // 姓名
};
class Student : public Person
{
protected:
	int _num; //學(xué)號
};
class Teacher : public Person
{
protected:
	int _id; // 編號
};
class Assistant : public Student, public Teacher
{
protected:
	string _Course; // 課程
};
int main()
{
	// 這樣會有二義性無法明確知道訪問的是哪一個
	Assistant a;
	//a._name = "peter";

	// 顯示的調(diào)用解決了二義性,但數(shù)據(jù)冗余了
	a.Student::_name = "蓋倫";
	a.Teacher::_name = "亞索";
	return 0;
}

虛繼承

虛擬繼承可以解決菱形繼承的二義性和數(shù)據(jù)冗余的問題。如上面的繼承關(guān)系,在Student和Teacher的繼承Person時使用虛擬繼承,即可解決問題。需要注意的是,虛擬繼承不要在其他地方去使用。

在菱形的腰部加上virtual關(guān)鍵字可以解決冗余。那虛繼承是怎么解決的呢?先來看看不用虛繼承的

class A {
public:
	int _a;
};

class B :  public A {
public:
	int _b;
};

class C :  public A {
public:
	int _c;
};
class D : public B, public C {
public:
	int _d;
};
int main()
{
	D d;
	d.B::_a = 1;
	d.C::_a = 2;
	d._b = 3;
	d._c = 4;
	d._d = 5;
	return 0;
}

我們可以通過內(nèi)存窗口來觀察對象成員的模型

菱形繼承帶來了二義性和數(shù)據(jù)冗余。

再來看看虛繼承的

虛繼承就解決了數(shù)據(jù)冗余和二義性,B和C中多了地址,在用內(nèi)存窗口看看這里的地址

2個指針叫虛基表指針指向虛基表,可以通過偏移量找到公共虛基類,此時A是在下面那為什么要找呢?

	D d;
	d.B::_a = 1;
	d.C::_a = 2;
	d._b = 3;
	d._c = 4;
	d._d = 5;
	B b = d;//把d賦給b,把d切過去那此時要怎么找到A呢?,所以就要用虛基表找

B類中各個成員在內(nèi)存中的分布:

通過偏移量找到虛基類。

還是不要用菱形繼承出現(xiàn)問題,虛繼承使得對象模型很復(fù)雜,并且會有效率的影響。

繼承的總結(jié)

  • C++語法復(fù)雜,其實多繼承就是一個體現(xiàn)。有了多繼承,就存在菱形繼承,有了菱形繼承就有菱形虛擬繼承,底層實現(xiàn)就很復(fù)雜。所以一般不建議設(shè)計出多繼承,一定不要設(shè)計出菱形繼承。否則在復(fù)雜度及性能上都有問題。
  • 多繼承可以認(rèn)為是C++的缺陷之一.

組合

繼承是建立了父類與子類的關(guān)系,是一種“是”的關(guān)系,例如白貓是貓,組合是“有”的關(guān)系實際盡量多去用組合。組合的耦合度低,代碼維護(hù)性好。不過繼承也有用武之地的,有些關(guān)系就適合繼承那就用繼承,另外要實現(xiàn)多態(tài),也必須要繼承。類之間的關(guān)系可以用繼承,可以用組合就用組合。

面試題

  • 什么是菱形繼承?菱形繼承的問題是什么?
    菱形繼承是多繼承的一種特殊繼承,兩個子類繼承同一個父類,而又有子類同時繼承這兩個子類??梢钥闯隽庑卫^承有數(shù)據(jù)冗余和二義性的問題。
  • 什么是菱形虛擬繼承?如何解決數(shù)據(jù)冗余和二義性的
    在菱形繼承的腰部加上virtual,通過虛基表指針和虛基表中的偏移量可以找到虛基類,只存1份
  • 繼承和組合的區(qū)別?什么時候用繼承?什么時候用組合?
    繼承是一種"是",組合是"有"的關(guān)系,父類和子類是的關(guān)系用繼承,是有的關(guān)系用組合。

以上就是C++繼承,由于作者水平有限,如有問題還請指出!

到此這篇關(guān)于C++繼承模式詳解的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • C語言實現(xiàn)三子棋游戲附注釋

    C語言實現(xiàn)三子棋游戲附注釋

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)三子棋游戲附注釋,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • C語言鏈表實現(xiàn)學(xué)生成績管理系統(tǒng)

    C語言鏈表實現(xiàn)學(xué)生成績管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語言鏈表實現(xiàn)學(xué)生成績管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-07-07
  • C語言實現(xiàn)鏈表與文件存取的示例代碼

    C語言實現(xiàn)鏈表與文件存取的示例代碼

    這篇文章主要和大家分享C語言實現(xiàn)鏈表與文件存取的示例代碼,可以實現(xiàn)建立鏈表,然后把鏈表數(shù)據(jù)存儲到文件中,然后把文件數(shù)據(jù)存儲到數(shù)組中并輸出,感興趣的可以學(xué)習(xí)一下
    2022-04-04
  • C++中int?main(int?argc,?char**?argv)的參數(shù)使用

    C++中int?main(int?argc,?char**?argv)的參數(shù)使用

    int?main(int?argc,?char**?argv)?是C和C++程序的入口點,其中argc和argv是用來接收從命令行傳遞給程序的參數(shù)的,本文就來介紹一下這兩個參數(shù)的含義,感興趣的可以了解一下的相關(guān)資料
    2024-01-01
  • C語言使用strcmp()函數(shù)比較兩個字符串的實現(xiàn)

    C語言使用strcmp()函數(shù)比較兩個字符串的實現(xiàn)

    這篇文章主要介紹了C語言使用strcmp()函數(shù)比較兩個字符串的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • C語言計算余數(shù)的圖文步驟

    C語言計算余數(shù)的圖文步驟

    在本篇文章里小編給大家整理了一篇關(guān)于C語言計算余數(shù)的圖文步驟內(nèi)容,有需要的朋友們可以參考下。
    2020-02-02
  • opencv3/C++圖像濾波實現(xiàn)方式

    opencv3/C++圖像濾波實現(xiàn)方式

    今天小編就為大家分享一篇opencv3/C++圖像濾波實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • 詳解設(shè)計模式中的Command命令模式及相關(guān)C++實現(xiàn)

    詳解設(shè)計模式中的Command命令模式及相關(guān)C++實現(xiàn)

    這篇文章主要介紹了詳解設(shè)計模式中的Command命令模式及相關(guān)C++實現(xiàn),命令模式強(qiáng)調(diào)調(diào)用操作的對象和操作的具體實現(xiàn)者之間的解耦,需要的朋友可以參考下
    2016-03-03
  • C語言數(shù)組實現(xiàn)三子棋應(yīng)用實例

    C語言數(shù)組實現(xiàn)三子棋應(yīng)用實例

    這篇文章主要為大家詳細(xì)介紹了C語言數(shù)組實現(xiàn)三子棋應(yīng)用實例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • c++動態(tài)規(guī)劃經(jīng)典算法

    c++動態(tài)規(guī)劃經(jīng)典算法

    動態(tài)規(guī)劃算法通常用于求解具有某種最優(yōu)性質(zhì)的問題。本文主要介紹了c++動態(tài)規(guī)劃經(jīng)典算法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08

最新評論