1.何為多態(tài)??多態(tài)的作用秀仲?
多態(tài)的概念:
一個接口,多種方法
多態(tài)的作用:
封裝可以是得代碼模塊化;繼承可以擴展已經(jīng)存在的代碼,都是為了代代碼重用;
多態(tài)的目的:接口重用
2.靜態(tài)聯(lián)編和動態(tài)聯(lián)編分別表示什么?
在編譯的時候能夠確定對象所調(diào)用的成員函數(shù)的地址則為靜態(tài)聯(lián)編;
動態(tài)聯(lián)編:指的是在程序運行的時候動態(tài)地進行,根據(jù)當時的情況來確定調(diào)用哪個同名函數(shù)(父類指針指向哪個子類,就調(diào)用哪個子類的同名函數(shù)),實際上是在運行時虛函數(shù)的實現(xiàn)
3.類中有虛函數(shù)的時候,類有什么變化?
當類中存在虛函數(shù)的時候,這個類大小就增加4個字節(jié),這4個字節(jié)是虛表指針,存放的是虛函數(shù)表的地址;
虛函數(shù)表其實是一個指針數(shù)組,它里面存放的其實是虛函數(shù)的地址
虛函數(shù)的幾個知識點:
先看一個例子:
#include "stdafx.h"
#include <ostream>
#include <iostream>
using namespace std;
class CClassA {
public:
virtual void fun_a() {
cout << "fun:cclassA,這是類A里面的函數(shù)" << endl;
}
};
class CClassB:public CClassA {
public:
virtual void fun_a() {
cout << "fun:cclassB,這是類B里面的函數(shù)" << endl;
}
};
int main()
{
CClassA objA, *pobjA;
CClassB objB;
pobjA = &objA;
pobjA->fun_a();
pobjA = &objB;//父類指針指向?qū)ο驜,下面這個函數(shù)就是B里面的對象!!!
pobjA->fun_a();
return 0;
}
幾個重點:
一個空類:一個字節(jié)
一個空類里面只有一個整型,此類大小:4個字節(jié)
- 定義基類的公有派生類
- 基類的公有派生類中重載該虛函數(shù)
- 重載該虛函數(shù)不是一般的重載,它要求函數(shù)名,返回類型,參數(shù)個數(shù),參數(shù)類型和順序完全相同
- 由于對虛函數(shù)進行了重載,派生類中的虛函數(shù)前的virtual關(guān)鍵字可以省略.
- 父類虛表和子類虛表是2個獨立的表(由于兩個類之間有繼承關(guān)系),故子類的虛表里面也有父類的虛函數(shù)指針,但是如果子類中有和父類中一樣的函數(shù)名,那么子類中的虛函數(shù)就覆蓋了子類虛表中所繼承的父類虛函數(shù)地址;
- 如果父類中有虛函數(shù),子類中無虛函數(shù),則子類也會生成一個虛表,(因為子類要繼承父類的虛函數(shù),但是虛函數(shù)又不能隨便放,只能生成一個虛表出來选浑!)(以上為個人理解,沒有太大偏差,想要理解的更深一點,需要查看相關(guān)文獻!!!)
4.純虛函數(shù)是一種特殊的虛函數(shù),是一種沒有具體實現(xiàn)的虛函數(shù)
例子:
class cclassA
{
virtual <函數(shù)類型><函數(shù)名>(<參數(shù)表>)=0炬搭;
}
含有純虛函數(shù)的類是抽象類,抽象類是不能定義對象的(不能實例化)
含有純虛函數(shù)的類---->抽象類------->不能定義對象!!!!!析構(gòu)函數(shù)為什么要推薦設(shè)計為虛函數(shù)??
首先,先了解一下構(gòu)造函數(shù)的調(diào)用順序:
基類構(gòu)造函數(shù)-->數(shù)據(jù)成員的構(gòu)造函數(shù)->派生類構(gòu)造函數(shù)
執(zhí)行派生類的析構(gòu)函數(shù),也需要調(diào)用基類以及子對象的析構(gòu)函數(shù),析構(gòu)順序如下:
派生類析構(gòu)函數(shù)-->數(shù)據(jù)成員類析構(gòu)函數(shù)----->基類的析構(gòu)函數(shù)
正常情況下,一個子類被釋放的時候,會主動調(diào)用其父類析構(gòu)函數(shù);使用父類指針指向子類對象的時候,只會析構(gòu)掉父類對象,如果此時子類里有堆空間內(nèi)存,則會造成內(nèi)存泄露翻伺!(內(nèi)存泄露是指使用malloc或者new申請內(nèi)存空間之后,沒有freeh或者delete掉,此時申請的那塊內(nèi)存仍然處于占用狀態(tài),稱為內(nèi)存泄露!)
故必須將析構(gòu)函數(shù)定義為虛函數(shù),這樣釋放父類指針的時候便會調(diào)用子類的析構(gòu)函數(shù),也會正常釋放掉子類的堆空間!
看下面一個例子:
#include "stdafx.h"
class Base
{
public:
Base() {
printf("base 父類構(gòu)造\n");
};
virtual ~Base() { //1th在此行加不加virtual有一定區(qū)別,而對于一般成員函數(shù)吼鳞,基類中有虛函數(shù)氢橙,則子類中對應(yīng)的成員函數(shù)不一定聲明為虛函數(shù)检诗,因為子類繼承了父類
printf("Base 父類析構(gòu)\n");
};
void fun() { //2th
printf("父類base-fun\n");
};
};
class son :public Base
{
public:
son() {
printf("son 子類構(gòu)造\n");
};
~son() { //**
printf("son 子類析構(gòu)\n");
}
void fun() { //3th
printf("子類son-fun\n");
};
};
int main()
{
Base *p; //父類指針
son *pobj = new son; //走到這一步,會先調(diào)用父類構(gòu)造,然后再調(diào)用子類構(gòu)造,new 出子類對象指針,從堆空間中分配出來的瓢剿,一般創(chuàng)建對象是在椃昊牛空間里面
p = pobj; //父類指針指向子類對象
p->fun(); //注意,這個例子里面函數(shù)不是虛函數(shù),所以調(diào)用的函數(shù)皆為父類里面的函數(shù)!!!!!
delete pobj; //通過父類指針釋放子類對象
return 0;
}
**處不管有沒有virtual這個關(guān)鍵詞,結(jié)果都如下:
(驗證了上面所說的:如果父類中有虛函數(shù),子類中無虛函數(shù),則子類也會生成一個虛表)
- 虛基類的目的是為了解決二義性問題,使用公共基類在其派生類對象中只產(chǎn)生一個基類子對象!!!!!