深入理解c++中virtual關(guān)鍵字
1.virtual關(guān)鍵字主要是什么作用?
c++中的函數(shù)調(diào)用默認(rèn)不適用動(dòng)態(tài)綁定。要觸發(fā)動(dòng)態(tài)綁定,必須滿足兩個(gè)條件:第一,指定為虛函數(shù);第二,通過基類類型的引用或指針調(diào)用。
由此可見,virtual主要主要是實(shí)現(xiàn)動(dòng)態(tài)綁定。
2.那些情況下可以使用virtual關(guān)鍵字?
virtual可用來定義類函數(shù)和應(yīng)用到虛繼承。
友元函數(shù) 構(gòu)造函數(shù) static靜態(tài)函數(shù) 不能用virtual關(guān)鍵字修飾;
普通成員函數(shù) 和析構(gòu)函數(shù) 可以用virtual關(guān)鍵字修飾;
3.virtual函數(shù)的效果
class GrandFather
{
public:
GrandFather() {}
virtual void fun()
{
cout << "GrandFather call function!" << endl;
}
};
class Father : public GrandFather
{
public:
Father() {}
void fun()
{
cout << "Father call function!" << endl;
}
};
class Son : public Father
{
public:
Son() {}
void fun()
{
cout << "Son call function!" << endl;
}
};
void print(GrandFather* father)
{
father->fun();
}
int _tmain(int argc, _TCHAR* argv[])
{
Father * pfather = new Son;
pfather->fun();
GrandFather * pgfather = new Father;
print(pgfather);
return 0;
}
輸出為 Son call function
Father call function
4.virtual的繼承性
只要基函數(shù)定義了virtual,繼承類的該函數(shù)也就具有virtual屬性
即 GrandFather Father Son同時(shí)定義virtual void fun()與GrandFather一個(gè)定義virtual void fun效果是一樣的
5.虛析構(gòu)函數(shù)
class GrandFather
{
public:
GrandFather() {}
virtual void fun()
{
cout << "GrandFather call function!" << endl;
}
~GrandFather()
{
cout << "GrandFather destruction!" << endl;
}
};
class Father : public GrandFather
{
public:
Father() {}
void fun()
{
cout << "Father call function!" << endl;
}
~Father()
{
cout << "Father destruction!" << endl;
}
};
class Son : public Father
{
public:
Son() {}
void fun()
{
cout << "Son call function!" << endl;
}
~Son()
{
cout << "Son destruction!" << endl;
}
};
void print(GrandFather* p)
{
p->fun();
}
int _tmain(int argc, _TCHAR* argv[])
{
Father * pfather = new Son;
delete pfather;
return 0;
}
以上代碼輸出:Father destruction!
GrandFather destruction!
執(zhí)行了Son的構(gòu)造函數(shù),沒執(zhí)行Son的析構(gòu)函數(shù),故把GrandFather的析構(gòu)函數(shù)設(shè)置為virtual
則輸出: Son destruction!
Father Destruction!
GrandFather destruction!
6. 純虛函數(shù)
純虛函數(shù)定義如下:
class GrandFather
{
public:
GrandFather() {}
virtual void fun() = 0
{
cout << "GrandFather call function!" << endl;
}
virtual ~GrandFather()
{
cout << "GrandFather destruction!" << endl;
}
};
純虛函數(shù)為后代類提供可覆蓋的接口,但這個(gè)類中的版本決不會(huì)調(diào)用。
含有(或繼續(xù))一個(gè)或多個(gè)純虛函數(shù)的類是抽象基類,抽象基類不能實(shí)例化!
繼承類只有重寫這個(gè)接口才能被實(shí)例化
7.虛繼承
虛繼承主要解決交叉繼承帶來的問題。這里給出一片參考文章c++虛繼承。
給一個(gè)例子如下
class GrandFather
{
public:
GrandFather() {}
void fun()
{
cout << "GrandFather call function!" << endl;
}
virtual ~GrandFather()
{
cout << "GrandFather destruction!" << endl;
}
};
class Father1 : public GrandFather
{
public:
Father1() {}
void fun()
{
cout << "Father call function!" << endl;
}
};
class Father2 : public GrandFather
{
public:
Father2() {}
void fun()
{
cout << "Father call function!" << endl;
}
};
class Son : public Father1, public Father2
{
public:
Son() {}
//void fun()
//{
// cout << "Son call function!" << endl;
//}
};
void print(GrandFather* p)
{
p->fun();
}
int _tmain(int argc, _TCHAR* argv[])
{
Son* son = new Son;
son->fun();
return 0;
}
編譯時(shí)會(huì)提示報(bào)錯(cuò)對(duì)fun的訪問不明確
如果Father1和Father2都用虛繼承繼承GrandFather類則可以解決這個(gè)問題
8. 構(gòu)造函數(shù)和析構(gòu)函數(shù)中的虛函數(shù)
如果在構(gòu)造函數(shù)或析構(gòu)函數(shù)中調(diào)用虛函數(shù),則運(yùn)行的是為構(gòu)造函數(shù)或析構(gòu)函數(shù)自身類型定義的版本
9.虛函數(shù)的實(shí)現(xiàn)機(jī)制
關(guān)于虛函數(shù)的實(shí)現(xiàn)機(jī)制,我們以后在介紹。
10.小結(jié)
關(guān)于virtual關(guān)鍵字的用法總結(jié)如上,有錯(cuò)誤或者總結(jié)不到位的情況請(qǐng)能幫本人指出!
11.例子
class classA
{
public:
classA()
{
clear();
}
virtual ~classA()
{
}
void clear()
{
memset(this , 0 , sizeof(*this));
}
virtual void func()
{
printf("func\n");
}
};
class classB : public classA
{
};
int main(void)
{
classA oa;
classB ob;
classA * pa0 = &oa;
classA * pa1 = &ob;
classB * pb = &ob;
oa.func(); // 1
ob.func(); // 2
pa0->func(); // 3
pa1->func(); // 4
pb->func(); // 5
return 0;
}
補(bǔ)充一個(gè)例子,這個(gè)程序輸出依次是
func
func
出錯(cuò)
func
func
談?wù)勎业睦斫猓?dāng)
classA oa;
oa.func();
不存在動(dòng)態(tài)調(diào)用的過程,所以func雖然是虛函數(shù),但是函數(shù)調(diào)用不通過虛表訪問,所以即使
memset(this , 0 , sizeof(*this));
找不到虛表地址也沒有關(guān)系
在執(zhí)行classB ob;的時(shí)候,注意memset的是classA的地址,所有ob的虛表是存在的
即是如下,通過指針或引用(動(dòng)態(tài)綁定)訪問oa的func函數(shù)(需要從虛表訪問),會(huì)出錯(cuò)
訪問ob的func和函數(shù),無論靜態(tài)訪問還是動(dòng)態(tài)訪問,都不會(huì)出錯(cuò)
當(dāng)把classB的代碼改成如下時(shí)
class classB : public classA
<PRE style="FONT-WEIGHT: bold" class=cpp name="code">{</PRE><PRE style="FONT-WEIGHT: bold" class=cpp name="code"> classB()
{
clear();
}
virtual ~classB()
{
}
void clear()
{
memset(this , 0 , sizeof(*this));
}</PRE><BR>
<PRE></PRE>
<PRE style="FONT-WEIGHT: bold" class=cpp name="code">};</PRE>輸出為
func
func
出錯(cuò)
出錯(cuò)
出錯(cuò)
上一篇:平衡二叉樹AVL操作模板
欄 目:C語言
本文標(biāo)題:深入理解c++中virtual關(guān)鍵字
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3767.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解約瑟夫環(huán)的數(shù)學(xué)優(yōu)化方法
- 01-10深入二叉樹兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10深入第K大數(shù)問題以及算法概要的詳解
- 01-10深入解析最長(zhǎng)公共子串
- 01-10c++中inline的用法分析
- 01-10深入理解鏈表的各類操作詳解


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