C++中抽象類和接口的區(qū)別介紹
1. 如果一個類B在語法上繼承(extend)了類A, 那么在語義上類B是一個類A.
2. 如果一個類B在語法上實現(xiàn)了(implement)接口I, 那么類B遵從接口I制定的協(xié)議.
使用abstract class的根本原因在于, 人們希望通過這樣的方式, 表現(xiàn)不同層次的抽象.
而interface的本質是一套協(xié)議. 在程序設計的發(fā)展中, 人們又發(fā)現(xiàn)接口可以用來表示對行為的抽象, 不過, 這只是interface的一種用法不是其本質.
理論結合實際才是最好的學習方式, 不過在這里, 我只想舉一些我見到過關于接口使用的反面教材:
1. 在接口中包含數(shù)據成員. 這幾乎肯定是錯的, 因為協(xié)議是規(guī)范是標準, 不應該跟具體實現(xiàn)有任何牽連, 也不應該給具體實現(xiàn)造成任何負擔.
2. C++中 delete 掉一個接口. 例如:
class IInterface()
{
Public:
Virtual ~IInterface(){};
…
}
Class ClassImpl : public IInterface
{
…
}
Int main()
{
IInterface* pInterface = new ClassImpl();
…
delete pInterface;
}
從語法的角度和語言自身的角度來看, 這是可行的, 而且只要將接口的析構函數(shù)設置為virtual, 就能避免內存泄漏. 但我要說, 這不是語法和語言的問題, 而是從根本上就錯了. 因為接口是一套協(xié)議, 一套規(guī)范, 并不是實現(xiàn). Delete 一個接口的代碼, 到底想要表達什么樣的語義? 如果一段代碼從語義上都說不通, 就不應該出現(xiàn)在程序中.
要在C++中表現(xiàn)接口的概念, 一種做法是這樣:
class IInterface
{
public:
virtual void DoSomething() = 0;
}
// 不應當有析構函數(shù), 因為從語義上說, 接口是不能delete的.
如果要delete, 只能delete一個類的實例:
Class A
{
Public:
Virtual ~A();
Public:
Virtual void DoSomething() = 0;
}
Class B : public A
{
…
}
Int main()
{
A* pA = new B();
…
Delete pA;
}
我們可以這樣做, 因為pA對應的是一個實例, 我們可以在A這一層將其銷毀.
先舉個例子,方便大家理解,然后從例子中抽象概括出結理論。
比如,一家生產門的公司,需要先定義好門的模板,以便能快速生產出各種規(guī)格的門。
這里的模板通常會有兩類模板:抽象類模板和接口模板。
抽象類模板:這個模板里面應該包含所有門都應該具有的共同屬性(如,門的形狀和顏色等)和共同行為(如,開門和關門)。
接口模板:有些門可能需要具有報警和指紋識別等功能,但這些功能又不是所有門必須具有的,所以像這樣的行為應該放在單獨的接口中。
有了上面的兩類模板,以后生產門就很方便了:利用抽象類模板和包含了報警功能的接口模板就能生產具有報警功能的門了。同理,利用抽象類模板和包含了指紋識別功能的接口模板就能生產具有指紋識別功能的門了。
總之:抽象類用來抽象自然界一些具有相似性質和行為的對象。而接口用來抽象行為的標準和規(guī)范,用來告訴接口的實現(xiàn)者必要按照某種規(guī)范去完成某個功能。
這是我自己的看法,歡迎大家和我探討這個問題。
相關文章
利用C語言模擬實現(xiàn)qsort,strcpy,strcat,strcmp函數(shù)
這篇文章主要為大家詳細介紹了如何通過C語言模擬實現(xiàn)qsort(采用冒泡的方式),strcpy,strcat,strcmp等函數(shù),文中的示例代碼講解詳細,感興趣的可以了解一下2022-11-11

