C++中的繼承模式深入詳解

前言

繼承是OOP設計中的重要概念。在C++語言中,派生類繼承基類有三種繼承方式:私有繼承(private)、保護繼承(protected)和公有繼承(public)。

一、繼承規則

繼承是C++中的重要特性,派生類可以訪問基類中的protected和public成員
先上代碼:

#include<iostream>
using namespace std;

class Base
{
	private:
			void func_pri();
	protected:
			void func_pro();
	public:
			void func_pub();
};

void Base::func_pri()
{
	cout << "private" << endl;
}
void Base::func_pro()
{
	cout << "protected" << endl;
}
void Base::func_pub()
{
	cout << "public" << endl;
	this->func_pri();
}

class Derive:public Base
{
	public:
		void function();
};

void Derive::function()
{
	//func_pri();調用private成員會產生編譯錯誤
	func_pro();
	func_pub();
}

int main()
{
	Derive* derive;
	derive=new Derive;
	derive->function();
	//output:
	//protected public private
	delete derive;
	return 0;
}

醬在這裡用一段簡單地代碼為大傢說明瞭繼承的規則。派生類隻能訪問基類的public和protected成員,無法訪問private成員;若要調用類中private成員,要在類內部使用this指針傳參(關於this指針的用法,醬在後續文章中會為大傢進一步整理)。這裡Derive是public Base,是公有繼承。
上表:

友元函數一直比較有爭議,有人認為它在一定程度上破壞瞭類的封裝性。關於友元函數的學習文章醬會在後續發出。

二、繼承類型

首先要確立一個概念:無論是公有、私有還是保護繼承,說到底它們都屬於繼承,滿足繼承的基本規則(如上表)。無論是private、protected還是public,對於直接向基類繼承的派生類來說幾乎沒有影響(即俗稱的“子代”而非“孫子代”)。上面的代碼class Derive:public Base換成private Base和protected Base,結果都是一樣的。
影響的是派生類的繼承,即“孫子代”。

1.保護繼承

先上代碼:

#include<iostream>
using namespace std;

class Base
{
	private:
			void func_pri();
	protected:
			void func_pro();
	public:
			void func_pub();
};

void Base::func_pri()
{
	cout << "private" << endl;
}
void Base::func_pro()
{
	cout << "protected" << endl;
}
void Base::func_pub()
{
	cout << "public" << endl;
	this->func_pri();
}

class Derive:protected Base
{
	public:
		void function();
};

void Derive::function()
{
	func_pro();
	func_pub();
}

class Derive_sec:public Derive
{
	public:
		void func_sec();
};

void Derive_sec::func_sec()
{
	func_pub();
	func_pro();
	//func_pri();調用這裡時會產生錯誤
}

int main()
{
	Derive_sec* derive;
	derive=new Derive_sec;
	derive->func_sec();
	delete derive;
	return 0;
}

這段代碼運行後的結果是public protected。
可以看出的是Derive_sec類(子二代)可以調用的是Base基類的protected和public成員函數,而無法調用private成員函數。

2.私有繼承

先上代碼:

#include<iostream>
using namespace std;

class Base
{
	private:
			void func_pri();
	protected:
			void func_pro();
	public:
			void func_pub();
};

void Base::func_pri()
{
	cout << "private" << endl;
}
void Base::func_pro()
{
	cout << "protected" << endl;
}
void Base::func_pub()
{
	cout << "public" << endl;
	this->func_pri();
}

class Derive:private Base
{
	public:
		void function();
};

void Derive::function()
{
	func_pro();
	func_pub();
}

class Derive_sec:public Derive
{
	public:
		void func_sec();
};

void Derive_sec::func_sec()
{
	//func_pub();調用時會產生錯誤
	//func_pro();調用時會產生錯誤
	//func_pri();調用這裡時會產生錯誤
	function();
}

int main()
{
	Derive_sec* derive;
	derive=new Derive_sec;
	derive->func_sec();
	delete derive;
	return 0;
}

通過結果來看,Derive_sec類(子二代)對Base基類中private、protected和public的成員函數均不能調用,但其仍然可以調用子一代的公有函數。

總結

在三種模式的繼承中,我們可以看出當子一代設置為公有繼承/保護繼承基類時,子二代對基類protected、public成員變量可以訪問,而對private變量不可訪問;當子一代設置為私有繼承基類時,子二代對基類private、protected、public成員變量均不能訪問.
我們可以得出這樣的結論:當基類和派生類具有直接繼承關系時,按照上文所述繼承規則即可;若派生類是經過對基類的多重繼承得來的,則要考慮之前的繼承模式。
private繼承模式在對類之間的封裝性有要求時,可以考慮使用。

到此這篇關於C++中的繼承模式深入詳解的文章就介紹到這瞭,更多相關C++中的繼承模式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: