C++ 多態和虛函數

虛函數實現多態

#include <iostream>
using namespace std;//基類People
class People{
public:virtual void display();  //聲明為虛函數
};
void People::display(){cout<<"無業游民。"<<endl;
}//派生類Teacher
class Teacher: public People{
public:virtual void display();  //聲明為虛函數
};
void Teacher::display(){cout<<"是一名教師"<<endl;
}int main(){People *p = new People();p -> display();p = new Teacher();p -> display();return 0;
}

結果:

無業游民。
是一名教師
?

有了虛函數,基類指針指向基類對象時就使用基類的成員(包括成員函數和成員變量),指向派生類對象時就使用派生類的成員。換句話說,基類指針可以按照基類的方式來做事,也可以按照派生類的方式來做事,它有多種形態,或者說有多種表現方式,我們將這種現象稱為多態(Polymorphism)

上面的代碼中,同樣是p->display();這條語句,當 p 指向不同的對象時,它執行的操作是不一樣的。同一條語句可以執行不同的操作,看起來有不同表現方式,這就是多態。

多態是面向對象編程的主要特征之一,C++中虛函數的唯一用處就是構成多態。

int main(){People p();Teacher t();People &rp = p;People &rt = t;rp.display();rt.display();return 0;
}

引用同樣可以多態

注意事項

1) 只需要在虛函數的聲明處加上?virtual 關鍵字,函數定義處可以加也可以不加。

2) 為了方便,你可以只將基類中的函數聲明為虛函數,這樣所有派生類中具有遮蔽關系的同名函數都將自動成為虛函數。

3) 當在基類中定義了虛函數時,如果派生類沒有定義新的函數來遮蔽此函數,那么將使用基類的虛函數。

4) 只有派生類的虛函數覆蓋基類的虛函數(函數原型相同)才能構成多態(通過基類指針訪問派生類函數)。例如基類虛函數的原型為virtual void func();,派生類虛函數的原型為virtual void func(int);,那么當基類指針 p 指向派生類對象時,語句p -> func(100);將會出錯,而語句p -> func();將調用基類的函數。

5) 構造函數不能是虛函數。對于基類的構造函數,它僅僅是在派生類構造函數中被調用,這種機制不同于繼承。也就是說,派生類不繼承基類的構造函數,將構造函數聲明為虛函數沒有什么意義。

6) 析構函數可以聲明為虛函數,而且有時候必須要聲明為虛函數,這點我們將在下節中講解。

純虛函數

在C++中,可以將虛函數聲明為純虛函數,語法格式為:

virtual 返回值類型 函數名 (函數參數) = 0;

純虛函數沒有函數體,只有函數聲明,在虛函數聲明的結尾加上=0,表明此函數為純虛函數。

最后的=0并不表示函數返回值為0,它只起形式上的作用,告訴編譯系統“這是純虛函數”。

包含純虛函數的類稱為抽象類(Abstract Class)。之所以說它抽象,是因為它無法實例化,也就是無法創建對象。原因很明顯,純虛函數沒有函數體,不是完整的函數,無法調用,也無法為其分配內存空間。

抽象類通常是作為基類,讓派生類去實現純虛函數。派生類必須實現純虛函數才能被實例化。

Line 是一個抽象類,也是最頂層的基類,在 Line 類中定義了兩個純虛函數 area() 和 volume()。

//線
class Line{
public:Line(float len);virtual float area() = 0;virtual float volume() = 0;
protected:float m_len;
};
Line::Line(float len): m_len(len){ }

在 Rec 類中,實現了 area() 函數;所謂實現,就是定義了純虛函數的函數體。但這時 Rec 仍不能被實例化,因為它沒有實現繼承來的 volume() 函數,volume() 仍然是純虛函數,所以 Rec 也仍然是抽象類。?

//矩形
class Rec: public Line{
public:Rec(float len, float width);float area();
protected:float m_width;
};
Rec::Rec(float len, float width): Line(len), m_width(width){ }
float Rec::area(){ return m_len * m_width; }

直到 Cuboid 類,才實現了 volume() 函數,才是一個完整的類,才可以被實例化。?

//長方體
class Cuboid: public Rec{
public:Cuboid(float len, float width, float height);float area();float volume();
protected:float m_height;
};
Cuboid::Cuboid(float len, float width, float height): Rec(len, width), m_height(height){ }
float Cuboid::area(){ return 2 * ( m_len*m_width + m_len*m_height + m_width*m_height); }
float Cuboid::volume(){ return m_len * m_width * m_height; }

它們的繼承關系為:Line --> Rec --> Cuboid
可以發現,Line 類表示“線”,沒有面積和體積,但它仍然定義了 area() 和 volume() 兩個純虛函數。這樣的用意很明顯:Line 類不需要被實例化,但是它為派生類提供了“約束條件”,派生類必須要實現這兩個函數,完成計算面積和體積的功能。
在實際開發中,你可以定義一個抽象基類,只完成部分功能,未完成的功能交給派生類去實現(誰派生誰實現)。這部分未完成的功能,往往是基類不需要的,或者在基類中無法實現的。雖然抽象基類沒有完成,但是卻強制要求派生類完成,這就是抽象基類的“霸王條款”。
?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/444059.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/444059.shtml
英文地址,請注明出處:http://en.pswp.cn/news/444059.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

圖的基本概念【數據結構】

序言 1對1的線性結構&#xff0c;一對多的樹二叉樹以及森林&#xff0c;第3種就是多對多的結構&#xff0c;也就是我們所要講到的圖的結構&#xff0c;圖形結構是數據結構當中最復雜的一種結構&#xff0c;圖形結構的特點就是在這個圖當中任意兩點之間都會有關系&#xff0c;這…

go語言一天入門(上)

第一個go程序 package mainimport "fmt"func main() {/* 這是我的第一個簡單的程序 */fmt.Println("Hello, World!") } 第一行代碼 package main 定義了包名。你必須在源文件中非注釋的第一行指明這個文件屬于哪個包&#xff0c;如&#xff1a;package m…

Jquery 全選,反選

<script src"../js/jquery-1.6.2.min.js" type"text/javascript"></script><script language"javascript" type"text/javascript">$(function(){$("#selectAll").click(function(){//全選$("#playLi…

go語言一天入門(下)

結構體 和c一樣 package mainimport "fmt"type Books struct {title stringauthor stringsubject stringbook_id int }func main() {// 創建一個新的結構體fmt.Println(Books{"Go 語言", "www.runoob.com", "Go 語言教程", 6495407}…

圖的遍歷算法【數據結構F】

圖的遍歷算法有哪兩種&#xff1f; 深度優先調度算法---------將圖結構看成是樹形結構&#xff0c;樹形結構的子圖直接是沒有交叉的&#xff0c;但是對于圖結構的樹形結構之間是有交叉的&#xff0c;類比于樹形結構的二叉樹&#xff0c;左指數和右指數都會相應的經歷三次&#…

go語言快速刷《程序員面試金典》(1)

實現一個算法&#xff0c;確定一個字符串 s 的所有字符是否全都不同。 一個數組統計是否有 func isUnique(astr string) bool {var arr[26] int;for _,ch:range astr{num:ch-aif(arr[num]1){return false}arr[num]}return true } 給定兩個字符串 s1 和 s2&#xff0c;請編寫一…

最小生成樹【數據結構】

前提 【1】網的最小生成樹&#xff0c;涉及到生成樹了那么就會有最小的權值在里面了 【2】對于一個圖來說生成樹是由多個的&#xff0c;并不是唯一的 【3】&#xff1a;廣度優先算法的遍歷是可以得到生成樹的&#xff0c;深度優先算法也是可以得到生成樹的 任意的一個聯通網&am…

go語言快速刷《程序員面試金典》(2)

字符串輪轉。給定兩個字符串s1和s2&#xff0c;請編寫代碼檢查s2是否為s1旋轉而成&#xff08;比如&#xff0c;waterbottle是erbottlewat旋轉后的字符串&#xff09;。 示例1 輸入&#xff1a;s1 "waterbottle", s2 "erbottlewat" 輸出&#xff1a;T…

廣義表的基本概念【數據結構】

實名廣義表與匿名廣義表的區別&#xff1a;對于匿名的廣義表的表示方法我們認為一對括號就是一個廣義表&#xff0c;里面的數據可以是廣義表也可以是 原子&#xff0c;對于有名字的廣義表&#xff0c;也就是大寫的字母我們可以直接認為大寫的就是廣義表的表示方法小練習----廣義…

go語言快速刷《程序員面試金典》(3)

編寫程序以 x 為基準分割鏈表&#xff0c;使得所有小于 x 的節點排在大于或等于 x 的節點之前。如果鏈表中包含 x&#xff0c;x 只需出現在小于 x 的元素之后(如下所示)。分割元素 x 只需處于“右半部分”即可&#xff0c;其不需要被置于左右兩部分之間。 示例: 輸入: head …

樹和二叉樹【數據結構】

基本概念 ADT的定義 基本操作 對比樹形結構和線性結構 基本術語以及注意事項-不能錯誤簡單的我以為 二叉樹是度數小于等于2的樹&#xff0c;而不是度為2的樹&#xff0c;一定要記住這個概念 小知識&#xff1a;二進制轉換成為十進制的方法名稱叫做位權求和法&#xff0c;用到…

leetcode557. 反轉字符串中的單詞 III python,處理字符串的神!

給定一個字符串&#xff0c;你需要反轉字符串中每個單詞的字符順序&#xff0c;同時仍保留空格和單詞的初始順序。 示例 1: 輸入: "Lets take LeetCode contest" 輸出: "steL ekat edoCteeL tsetnoc" 注意&#xff1a;在字符串中&#xff0c;每個單詞由…

數據庫2.1.1mysql的特點

在mysql5.1當中&#xff0c;mysqlab公司引入了新的插件式存儲引擎體系結構&#xff0c;也許將存儲引擎加載到正在運行的mysql服務器當中&#xff0c;使用mysql插件是存儲引擎體系結構允許數據庫用戶為特定的應用需求選擇專門的存儲引擎&#xff0c;完全不需要管理任何特殊的應用…

leetcode369. 給單鏈表加一

用一個 非空 單鏈表來表示一個非負整數&#xff0c;然后將這個整數加一。 你可以假設這個整數除了 0 本身&#xff0c;沒有任何前導的 0。 這個整數的各個數位按照 高位在鏈表頭部、低位在鏈表尾部 的順序排列。 示例: 輸入: [1,2,3] 輸出: [1,2,4] 思路&#xff1a; hel…

MySQL常見的兩種存儲引擎:MyISAM與InnoDB的愛恨情仇

一 MyISAM 1.1 MyISAM簡介 MyISAM是MySQL的默認數據庫引擎&#xff08;5.5版之前&#xff09;&#xff0c;由早期的 ISAM &#xff08;Indexed Sequential Access Method&#xff1a;有索引的順序訪問方法&#xff09;所改良。雖然性能極佳&#xff0c;而且提供了大量的特性&a…

leetcode193. 有效電話號碼 正則了解一下

給定一個包含電話號碼列表&#xff08;一行一個電話號碼&#xff09;的文本文件 file.txt&#xff0c;寫一個 bash 腳本輸出所有有效的電話號碼。 你可以假設一個有效的電話號碼必須滿足以下兩種格式&#xff1a; (xxx) xxx-xxxx 或 xxx-xxx-xxxx。&#xff08;x 表示一個數字…

leetcode258. 各位相加

給定一個非負整數 num&#xff0c;反復將各個位上的數字相加&#xff0c;直到結果為一位數。 示例: 輸入: 38 輸出: 2 解釋: 各位相加的過程為&#xff1a;3 8 11, 1 1 2。 由于 2 是一位數&#xff0c;所以返回 2。 進階: 你可以不使用循環或者遞歸&#xff0c;且在 O(…

leetcode412. Fizz Buzz

寫一個程序&#xff0c;輸出從 1 到 n 數字的字符串表示。 1. 如果 n 是3的倍數&#xff0c;輸出“Fizz”&#xff1b; 2. 如果 n 是5的倍數&#xff0c;輸出“Buzz”&#xff1b; 3.如果 n 同時是3和5的倍數&#xff0c;輸出 “FizzBuzz”。 示例&#xff1a; n 15, 返…

leetcode359. 日志速率限制器

請你設計一個日志系統&#xff0c;可以流式接收日志以及它的時間戳。 該日志會被打印出來&#xff0c;需要滿足一個條件&#xff1a;當且僅當日志內容 在過去的 10 秒鐘內沒有被打印過。 給你一條日志的內容和它的時間戳&#xff08;粒度為秒級&#xff09;&#xff0c;如果這…

怎樣提高WebService性能大數據量網絡傳輸處理(轉)

1. 直接返回DataSet對象 特點&#xff1a;通常組件化的處理機制&#xff0c;不加任何修飾及 處理&#xff1b; 優點&#xff1a;代碼精減、易于處理&#xff0c;小數據量處理較快&#xff1b; 缺點&#xff1a;大數據量的傳遞處理慢&#xff0c;消耗網絡資源&#xff1b; 建議&…