C/C++中虛函數(shù)詳解及其作用介紹
概述
虛函數(shù) (virtual function) 指可以被子類繼承和覆蓋的函數(shù).

使用方法
基類聲明成員函數(shù)為虛函數(shù)的方法:
virtual [類型] 函數(shù)名([參數(shù)表列])
注: 在類外定義虛函數(shù)時(shí), 不需再加 virtual.
虛函數(shù)的特點(diǎn):
- 提高程序擴(kuò)充性: 派生類根據(jù)需要可以進(jìn)行函數(shù)覆蓋
- 成員函數(shù)被聲明為虛數(shù)后, 其派生類中覆蓋函數(shù)自動(dòng)稱為虛函數(shù)
- 若虛函數(shù)在派生類中未重新定義, 則派生類簡(jiǎn)單繼承其直接基類的虛函數(shù)
- 指向基類的指針, 當(dāng)指向派生類對(duì)象時(shí), 可以嗲用派生類的方法
關(guān)聯(lián)
通過(guò)關(guān)聯(lián) (binding), 我們可以把一個(gè)標(biāo)識(shí)符和一個(gè)存儲(chǔ)地址聯(lián)系起來(lái), 或者把一個(gè)函數(shù)名與一個(gè)類對(duì)象捆綁在一起.

靜態(tài)關(guān)聯(lián)
靜態(tài)關(guān)聯(lián) (static binding) 指通過(guò)對(duì)象名調(diào)用虛函數(shù). 在編譯時(shí)即可確定其調(diào)用的虛函數(shù)屬于哪一類
動(dòng)態(tài)關(guān)聯(lián)
動(dòng)態(tài)關(guān)聯(lián) (dynamic binding) 是指通過(guò)基類指針與虛函數(shù), 在運(yùn)行階段確定關(guān)聯(lián)關(guān)系. 動(dòng)態(tài)關(guān)聯(lián)提供動(dòng)態(tài)的多態(tài)性, 即運(yùn)行階段的多態(tài)性.
案例1
未使用虛函數(shù)
Square 類:
#ifndef PROJECT6_SQUARE_H
#define PROJECT6_SQUARE_H
class Square {
protected:
int length;
public:
Square(int l) : length(l) {};
int area() const {
return length *length;
}
};
#endif //PROJECT6_SQUARE_H
Rectangle 類:
#ifndef PROJECT6_RECTANGLE_H
#define PROJECT6_RECTANGLE_H
#include "Square.h"
class Rectangle : public Square{
private:
int height;
public:
Rectangle(int l, int h) : Square(l), height(h) {};
int area() const {
return Square::area() * 2 + length * height * 4; // 兩個(gè)底加四個(gè)邊
}
};
#endif //PROJECT6_RECTANGLE_H
main:
#include <iostream>
#include "Square.h"
#include "Rectangle.h"
using namespace std;
int main() {
// 創(chuàng)建對(duì)象
Square s1(2), *pt;
Rectangle r1(3, 3);
pt = &s1;
cout << pt->area() << endl;
pt = &r1;
cout << pt->area() << endl;
return 0;
}
輸出結(jié)果:
4
9 // 輸出的是底面積
此時(shí)調(diào)用的是 Square 類的area()函數(shù).
使用虛擬類
Square 類:
#ifndef PROJECT6_SQUARE_H
#define PROJECT6_SQUARE_H
class Square {
protected:
int length;
public:
Square(int l) : length(l) {};
virtual int area() const {
return length *length;
}
};
#endif //PROJECT6_SQUARE_H
Rectangle 類:
#ifndef PROJECT6_RECTANGLE_H
#define PROJECT6_RECTANGLE_H
#include "Square.h"
class Rectangle : public Square{
private:
int height;
public:
Rectangle(int l, int h) : Square(l), height(h) {};
int area() const {
return Square::area() * 2 + length * height * 4; // 兩個(gè)底加四個(gè)邊
}
};
#endif //PROJECT6_RECTANGLE_H
main:
#include <iostream>
#include "Square.h"
#include "Rectangle.h"
using namespace std;
int main() {
// 創(chuàng)建對(duì)象
Square s1(2), *pt;
Rectangle r1(3, 3);
pt = &s1;
cout << pt->area() << endl;
pt = &r1;
cout << pt->area() << endl;
return 0;
}
輸出結(jié)果:
4
54 // 長(zhǎng)方體的面積
此時(shí)調(diào)用的是 Rectangle 類的area()函數(shù).
案例2
Animal 類:
#ifndef PROJECT6_ANIMAL_H
#define PROJECT6_ANIMAL_H
#include <iostream>
using namespace std;
class Animal {
public:
virtual void bark(){
cout << "咋叫?" << endl;
}
};
#endif //PROJECT6_ANIMAL_H
Dog 類:
#ifndef PROJECT6_DOG_H
#define PROJECT6_DOG_H
#include "Animal.h"
class Dog : public Animal{
public:
void bark() {
cout << "汪汪!" << endl;
}
};
#endif //PROJECT6_DOG_H
Cat 類:
#ifndef PROJECT6_CAT_H
#define PROJECT6_CAT_H
#include "Animal.h"
class Cat : public Animal{
public:
void bark() {
cout << "喵喵!" << endl;
}
};
#endif //PROJECT6_CAT_H
Pig 類:
#ifndef PROJECT6_PIG_H
#define PROJECT6_PIG_H
#include "Animal.h"
class Pig : public Animal {
public:
void bark() {
cout << "哼哼!" << endl;
}
};
#endif //PROJECT6_PIG_H
main:
#include <iostream>
#include "Animal.h"
#include "Dog.h"
#include "Cat.h"
#include "Pig.h"
using namespace std;
int main() {
// 創(chuàng)建對(duì)象
Animal a, *pt;
Dog d;
Cat c;
Pig p;
pt = &a;
pt -> bark(); // 調(diào)用基類的bark()
pt = &d;
pt -> bark(); // 調(diào)用狗的bark()
pt = &c;
pt -> bark(); // 調(diào)用貓的bark()
pt = &p;
pt -> bark(); // 調(diào)用豬的bark()
return 0;
}
輸出結(jié)果:
咋叫?
汪汪!
喵喵!
哼哼!
總結(jié)
虛函數(shù)只能是類的成員函數(shù), 而不能將類外的普通函數(shù)聲明為虛函數(shù). 虛函數(shù)的作用是允許在派生類中對(duì)基類的虛函數(shù)重新定義 (函數(shù)覆蓋), 只能用于類的繼承層次結(jié)構(gòu)中.
虛函數(shù)能有效減少空間開(kāi)銷. 當(dāng)一個(gè)類帶有虛函數(shù)時(shí), 編譯系統(tǒng)會(huì)為該類構(gòu)造一個(gè)虛函數(shù)表 (一個(gè)指針數(shù)組), 用于存放每個(gè)虛函數(shù)的入口地址.
什么時(shí)候應(yīng)該使用虛函數(shù):
- 判斷成員函數(shù)所在的類是不是基類, 非基類無(wú)需使用虛函數(shù)
- 成員函數(shù)在類被繼承后有沒(méi)有可能被更改的功能, 如果希望修改成員函數(shù)功能, 一般在基類中將其聲明為虛函數(shù)
- 我們會(huì)通過(guò)對(duì)象名還是基類指針訪問(wèn)成員函數(shù), 如果通過(guò)基類指針過(guò)引用去訪問(wèn), 則應(yīng)當(dāng)聲明為虛函數(shù)
有時(shí)候在定義虛函數(shù)的時(shí)候, 我們無(wú)需定義其函數(shù)體. 它的作用只是定義了一個(gè)虛函數(shù)名, 具體的功能留給派生類去添加, 也就是純虛函數(shù). 例如我們?cè)谏厦娴?Animal 類的bark()函數(shù)就應(yīng)該聲明為純虛函數(shù), 因?yàn)?Animal 為基類, 定義bark()函數(shù)實(shí)體并無(wú)意義.
相關(guān)文章
使用C/C++讀取matlab中.mat格式數(shù)據(jù)的操作
這篇文章給大家介紹了使用C/C++讀取matlab中.mat格式數(shù)據(jù)的操作,文中通過(guò)圖文結(jié)合的方式介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-12-12
C語(yǔ)言中的字符型數(shù)據(jù)與ASCII碼表
這篇文章主要介紹了C語(yǔ)言中的字符型數(shù)據(jù)與ASCII碼表,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
C語(yǔ)言 結(jié)構(gòu)體數(shù)組詳解及示例代碼
本文主要介紹C語(yǔ)言 結(jié)構(gòu)體數(shù)組,這里整理了相關(guān)資料及簡(jiǎn)單示例代碼,以便大家學(xué)習(xí)參考,有興趣的小伙伴可以看下2016-08-08
C語(yǔ)言 完整游戲項(xiàng)目推箱子詳細(xì)代碼
經(jīng)典的推箱子是一個(gè)的古老游戲,目的是在訓(xùn)練你的邏輯思考能力。在一個(gè)狹小的倉(cāng)庫(kù)中,要求把木箱放到指定的位置,稍不小心就會(huì)出現(xiàn)箱子無(wú)法移動(dòng)或者通道被堵住的情況,所以需要巧妙的利用有限的空間和通道,合理安排移動(dòng)的次序和位置,才能順利的完成任務(wù)2021-11-11
C++面向?qū)ο笳Z(yǔ)言自制多級(jí)菜單功能實(shí)現(xiàn)代碼
菜單類主要負(fù)責(zé)菜單的創(chuàng)建、修改、刪除,是包含菜單結(jié)構(gòu)組織和響應(yīng)函數(shù)的模型,用戶擁有充分的自主性,可根據(jù)需要自定義菜單顯示和響應(yīng)函數(shù),這篇文章主要介紹了C++面向?qū)ο笳Z(yǔ)言自制多級(jí)菜單,需要的朋友可以參考下2024-06-06

