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