C++面向?qū)ο笾鄳B(tài)的實(shí)現(xiàn)和應(yīng)用詳解
前言
本文主要給大家介紹的是關(guān)于C++面向?qū)ο笾鄳B(tài)的實(shí)現(xiàn)和應(yīng)用的相關(guān)內(nèi)容,分享出來(lái)供大家參考學(xué)習(xí),下面話(huà)不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧。
多態(tài)
大家應(yīng)該都聽(tīng)過(guò)C++三大特性之一多態(tài),那么什么多態(tài)呢?多態(tài)有什么用?通俗一點(diǎn)來(lái)講->
多態(tài)性可以簡(jiǎn)單地概括為“一個(gè)接口,多種方法”,程序在運(yùn)行時(shí)才決定調(diào)用的函數(shù),它是面向?qū)ο缶幊填I(lǐng)域的核心概念。當(dāng)多態(tài)應(yīng)用形參類(lèi)型的時(shí)候,可以接受更多的類(lèi)型。當(dāng)多態(tài)用于返回值類(lèi)型的時(shí)候,可以返回更多類(lèi)型的數(shù)據(jù)。多態(tài)可以讓你的代碼擁有更好的擴(kuò)展性。
多態(tài)分兩種分別為靜態(tài)多態(tài)和動(dòng)態(tài)多態(tài):
- 靜態(tài)多態(tài):靜態(tài)多態(tài)就是重載,因?yàn)槭窃诰幾g期決議確定,所以稱(chēng)為靜態(tài)多態(tài)。
- 動(dòng)態(tài)多態(tài):動(dòng)態(tài)多態(tài)就是通過(guò)繼承重寫(xiě)基類(lèi)的虛函數(shù)實(shí)現(xiàn)的多態(tài),因?yàn)槭窃谶\(yùn)行時(shí)決議確定,所以稱(chēng)為動(dòng)態(tài)多態(tài)。
而我們主要今天來(lái)看動(dòng)態(tài)多態(tài)的問(wèn)題。比如我們來(lái)看下面的代碼,就是簡(jiǎn)單的動(dòng)態(tài)多態(tài):
class Person
{
public:
virtual void BuyTickets()
{
cout << " 買(mǎi)票" << endl;
}
protected:
string _name; // 姓名
};
class Student : public Person
{
public:
virtual void BuyTickets()
{
cout << " 買(mǎi)票-半價(jià) " << endl;
}
protected:
int _num; //學(xué)號(hào)
};
void Fun(Person& p)
{
p.BuyTickets();
}
void Test()
{
Person p;
Student s;
Fun(p);
Fun(s);
}
int main()
{
Test();
system("pause");
return 0;
}
構(gòu)成多態(tài)的四大條件: (缺一不可)
1.不在同一作用域(分別在父類(lèi)和子類(lèi))
2.函數(shù)名相等/參數(shù)相等/返回值相同/(協(xié)變除外)
3.基類(lèi)函數(shù)必須有virtual關(guān)鍵字
4.訪(fǎng)問(wèn)修飾符可以不同
具體多態(tài)是如何實(shí)現(xiàn)的?? 這里我們先從虛函數(shù)表這個(gè)知識(shí)點(diǎn)講起,每一個(gè)帶有虛函數(shù)的對(duì)象都會(huì)有一個(gè)虛函數(shù)表,虛函數(shù)表里存的是函數(shù)指針,然后調(diào)用的時(shí)候,指針回去虛函數(shù)表里面訪(fǎng)問(wèn)查找。對(duì)于這個(gè)知識(shí)點(diǎn)我的另外一個(gè)博客很詳細(xì)的講解到,大家可以先看看這個(gè)://www.jb51.net/article/123308.htm
然后我們來(lái)了解一下重寫(xiě)是什么東西?
重寫(xiě)的過(guò)程
如果這塊還是不理解,你可以看我專(zhuān)門(mén)寫(xiě)虛函數(shù)那片博客,仔細(xì)看一定會(huì)看懂的.
接下來(lái)多態(tài)的原理我們就明白了吧. 發(fā)生重寫(xiě)之后,下一次父類(lèi)指針指向我調(diào)用fun()函數(shù)的時(shí)候,它調(diào)用到的就是子類(lèi)的fun()函數(shù),其實(shí)多態(tài)就是這么簡(jiǎn)單,只要理解重寫(xiě)就理解多態(tài). 虛函數(shù)表是我們必須掌握的一個(gè)知識(shí)點(diǎn).
通過(guò)匯編來(lái)分析多態(tài)的實(shí)現(xiàn)
好了,我們繼續(xù)往下走,剛剛我們從虛函數(shù)表這方面,探究了多態(tài)的實(shí)現(xiàn),現(xiàn)在我們?cè)購(gòu)膮R編的角度再來(lái)看多態(tài)是如何實(shí)現(xiàn)的。
我們來(lái)看一段新的代碼:
class Person
{
public:
virtual void BuyTickets()
{
cout << " 買(mǎi)票" << endl;
}
protected:
string _name; // 姓名
};
class Student : public Person
{
public:
virtual void BuyTickets()
{
cout << " 買(mǎi)票-半價(jià) " << endl;
}
protected:
int _num; //學(xué)號(hào)
};
void Fun(Person& p)
{
p.BuyTickets();
}
void Test()
{
Person p;
Student q;
Person* ptr = &q;
p.BuyTickets();
ptr->BuyTickets();
}
int main()
{
Test();
system("pause");
return 0;
}
打開(kāi)我們的反匯編窗口:
這里我們看到用指向子類(lèi)的父類(lèi)類(lèi)型指針調(diào)用BuyTickets函數(shù)和直接用對(duì)象調(diào)用匯編代碼相差巨大,一個(gè)只有2句話(huà),一個(gè)那么長(zhǎng),這是因?yàn)樵诎l(fā)生多態(tài)時(shí)當(dāng)你用指針調(diào)用時(shí),系統(tǒng)不知道你要用哪一個(gè)函數(shù),因?yàn)檫@里有多態(tài)現(xiàn)象,所以系統(tǒng)只能老實(shí)的去虛函數(shù)表里查找,所以才會(huì)有這么多的代碼,接下來(lái)我們來(lái)解釋一下這些匯編,來(lái)看看系統(tǒng)是調(diào)用虛表的。
這里我們就關(guān)心到了那四個(gè)紅色的句子,可以看到這里一直都是想講虛函數(shù)表的地址傳給系統(tǒng),然后再傳this指針,就可以調(diào)用哪個(gè)函數(shù)了。藍(lán)色的就是一個(gè)小知識(shí)~ 知道有這么個(gè)東西就好了.
虛函數(shù)是在基類(lèi)中定義的,目的是不確定它的派生類(lèi)的具體行為。例:
- 定義一個(gè)基類(lèi):class Animal//動(dòng)物。它的函數(shù)為breathe()//呼吸。
- 再定義一個(gè)類(lèi)class Fish//魚(yú) 。它的函數(shù)也為breathe()
- 再定義一個(gè)類(lèi)class Sheep //羊。它的函數(shù)也為breathe()
- 為了簡(jiǎn)化代碼,將Fish,Sheep定義成基類(lèi)Animal的派生類(lèi)。
然而Fish與Sheep的breathe不一樣,一個(gè)是在水中通過(guò)水來(lái)呼吸,一個(gè)是直接呼吸空氣。所以基類(lèi)不能確定該如何定義breathe,所以在基類(lèi)中只定義了一個(gè)virtual breathe,它是一個(gè)空的虛函數(shù)。具本的函數(shù)在子類(lèi)中分別定義。程序一般運(yùn)行時(shí),找到類(lèi),如果它有基類(lèi),再找它的基類(lèi),最后運(yùn)行的是基類(lèi)中的函數(shù),這時(shí),它在基類(lèi)中找到的是virtual標(biāo)識(shí)的函數(shù),它就會(huì)再回到子類(lèi)中找同名函數(shù)。派生類(lèi)也叫子類(lèi)?;?lèi)也叫父類(lèi)。這就是虛函數(shù)的產(chǎn)生,和類(lèi)的多態(tài)性(breathe)的體現(xiàn)。
一般情況下(沒(méi)有涉及virtual函數(shù)),當(dāng)我們用一個(gè)指針/引用調(diào)用一個(gè)函數(shù)的時(shí)候,被調(diào)用的函數(shù)是取決于這個(gè)指針/引用的類(lèi)型。即如果這個(gè)指針/引用是基類(lèi)對(duì)象的指針/引用就調(diào)用基類(lèi)的方法;如果指針/引用是派生類(lèi)對(duì)象的指針/引用就調(diào)用派生類(lèi)的方法,當(dāng)然如果派生類(lèi)中沒(méi)有此方法,就會(huì)向上到基類(lèi)里面去尋找相應(yīng)的方法。這些調(diào)用在編譯階段就確定了。
當(dāng)設(shè)計(jì)到多態(tài)性的時(shí)候,采用了虛函數(shù)和動(dòng)態(tài)綁定,此時(shí)的調(diào)用就不會(huì)在編譯時(shí)候確定而是在運(yùn)行時(shí)確定。不在單獨(dú)考慮指針/引用的類(lèi)型而是看指針/引用的對(duì)象的類(lèi)型來(lái)判斷函數(shù)的調(diào)用,根據(jù)對(duì)象中虛指針指向的虛表中的函數(shù)的地址來(lái)確定調(diào)用哪個(gè)函數(shù)。
現(xiàn)在我們來(lái)一個(gè)小練習(xí):
#include<iostream>
#include<Windows.h>
using namespace std;
class A
{
public:
void foo()
{
printf("1\n");
}
virtual void fun()
{
printf("2\n");
}
};
class B : public A
{
public:
void foo()
{
printf("3\n");
}
void fun()
{
printf("4\n");
}
};
int main(void)
{
A a;
B b;
A *p = &a;
p->foo();
p->fun();
p = &b;
p->foo();
p->fun();
system("pause");
return 0;
}
這道題的運(yùn)行結(jié)果分別是 1 2 1 4,,現(xiàn)在我們來(lái)分析為什么?
首先當(dāng)一個(gè)父類(lèi)類(lèi)型指針指向父類(lèi)時(shí),我們應(yīng)該知道這里沒(méi)有多態(tài),該怎么調(diào)用就怎么調(diào)用,所以調(diào)用了父類(lèi)里面的foo函數(shù)和fun函數(shù)?,F(xiàn)在我們重點(diǎn)來(lái)看后面這個(gè),現(xiàn)在B繼承了A,我們先判斷這里是否有多態(tài)現(xiàn)象(1.父類(lèi)和子類(lèi)是否有重寫(xiě)現(xiàn)象 2.是否有父類(lèi)類(lèi)型的指針指向子類(lèi)),現(xiàn)在很明顯子類(lèi)的fun函數(shù)重寫(xiě)了父類(lèi)的fun函數(shù),所以現(xiàn)在p->fun()調(diào)用的就是子類(lèi)的fun函數(shù),然后foo函數(shù),根本不構(gòu)成多態(tài),所以這里指針類(lèi)型是什么那個(gè)對(duì)象就按那個(gè)對(duì)象調(diào)用??偨Y(jié)一下當(dāng)你碰到關(guān)于繼承的問(wèn)題,首先判斷它里面是否有多態(tài)現(xiàn)象,如果沒(méi)有那就根據(jù)指針/引用類(lèi)型調(diào)用。如果有多態(tài)的話(huà),一定要注意根據(jù)指針/引用的指向?qū)ο笈袛唷?/p>
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)我們的支持。
上一篇:C語(yǔ)言實(shí)現(xiàn)最簡(jiǎn)單的剪刀石頭布小游戲示例
欄 目:C語(yǔ)言
下一篇:C語(yǔ)言如何利用異或進(jìn)行兩個(gè)值的交換詳解
本文標(biāo)題:C++面向?qū)ο笾鄳B(tài)的實(shí)現(xiàn)和應(yīng)用詳解
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1202.html
您可能感興趣的文章
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見(jiàn)的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類(lèi)算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10delphi制作wav文件的方法
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改


