【C++】深入理解C++虛函數與純虛函數

文章目錄

  • 一、虛函數(Virtual Function)
    • 1.1 定義和作用
    • 1.2 實現原理
    • 1.3 示例代碼
    • 1.4 虛函數的重寫
      • 定義
      • 規則
      • 注意事項
      • 示例
    • 1.5 基類和派生類的虛函數表
      • **示例理解**
  • 二、純虛函數(Pure Virtual Function)
    • 2.1 定義和作用
    • 2.2 示例代碼
  • 三、總結

在C++面向對象編程中,多態性是其三大特性之一(封裝、繼承和多態)。為了實現多態性,C++引入了虛函數(Virtual Function)和純虛函數(Pure Virtual Function)的概念。本文將深入探討虛函數和純虛函數的原理和應用,幫助讀者更好地理解它們在C++中的作用。

一、虛函數(Virtual Function)

1.1 定義和作用

虛函數是在基類中使用關鍵字 virtual 聲明的成員函數,它允許派生類對其進行重寫(Override),實現運行時多態。當通過基類指針或引用調用虛函數時,實際調用的是對象類型對應的派生類中的函數,這個過程稱為動態綁定(Dynamic Binding)或晚綁定(Late Binding)。

1.2 實現原理

虛函數的實現原理基于虛函數表(Virtual Table,簡稱VTable)。每個使用虛函數的類都有一個虛函數表,該表是一個函數指針數組,存儲了指向類的虛函數的指針。類的每個實例都包含一個指向其虛函數表的指針(vptr),通過這個指針可以找到并調用正確的虛函數實現。

當派生類覆蓋(重寫)基類的虛函數時,派生類的虛函數表中相應位置的函數指針會被更新為指向派生類中的函數。如果派生類沒有重寫虛函數,則派生類的虛函數表中會保留指向基類虛函數的指針。

1.3 示例代碼

#include <iostream>
using namespace std;class Base {
public:virtual void show() {cout << "Base class show" << endl;}
};class Derived : public Base {
public:void show() override {cout << "Derived class show" << endl;}
};int main() {Base* b = new Derived();b->show(); // 輸出:Derived class showdelete b;return 0;
}

1.4 虛函數的重寫

虛函數的重寫(Override)是面向對象編程中實現多態性的一種方式。虛函數允許派生類根據需要改變或擴展基類中的行為。這里,我們將詳細探討虛函數的重寫,包括它的定義、規則以及一些注意事項。

定義

虛函數重寫指的是派生類中提供一個函數版本,該版本與基類中具有相同名稱、相同返回類型和相同參數列表的虛函數相匹配。通過這種方式,派生類可以提供自己特定的實現,替換或擴展基類的行為。

規則

  1. 函數簽名必須匹配:要重寫基類中的虛函數,派生類中的函數必須具有相同的名稱、返回類型和參數列表。
  2. 基類函數必須是虛函數:只有虛函數可以被重寫。如果基類中的函數不是虛函數,派生類中相同簽名的函數會隱藏(而非重寫)基類中的函數。
  3. 訪問權限可以不同:虛函數在派生類中的訪問級別(public、protected、private)可以與基類中的不同,但這會影響到函數的訪問性。
  4. 使用 override 關鍵字(C++11及以上):雖然不是強制的,但建議在派生類中重寫虛函數時使用 override 關鍵字,這有助于編譯器檢查函數簽名是否正確匹配,避免潛在的錯誤。

注意事項

  • 析構函數應該是虛的:如果一個類有可能被繼承,并且通過基類指針來刪除派生類對象,那么基類的析構函數應該是虛的。這確保了通過基類指針刪除派生類對象時,能夠正確地調用派生類的析構函數。
  • 構造函數不能是虛函數:在C++中,構造函數不能被聲明為虛函數。因為構造函數是用來創建對象的,而虛函數的調用需要通過對象的虛函數表,這在對象構造階段還未完全建立。
  • 使用 final 關鍵字防止進一步重寫:在某些情況下,你可能希望禁止進一步重寫某個虛函數。C++11引入了final關鍵字,可以用來阻止派生類重寫特定的虛函數。

示例

#include <iostream>class Base {
public:virtual void print() const {std::cout << "Base class print function" << std::endl;}virtual ~Base() {} // 虛析構函數
};class Derived : public Base {
public:void print() const override { // 使用override確保正確重寫std::cout << "Derived class print function" << std::endl;}
};int main() {Base* b = new Derived();b->print(); // 輸出:Derived class print functiondelete b; // 正確調用派生類析構函數return 0;
}

在上述示例中,Derived 類重寫了 Base 類中的 print 函數,并且基類的析構函數被聲明為虛函數,確保了通過基類指針刪除派生類對象時能夠正確調用派生類的析構函數。

通過理解和正確應用虛函數的重寫,可以充分利用C++的多態性,設計出靈活且易于維護的面向對象程序。

1.5 基類和派生類的虛函數表

當涉及到繼承時,虛函數表(vtable)的處理方式會稍微復雜一些,但關鍵點在于每個類都有自己的虛函數表,而不是只有一個。這意味著,如果有派生類繼承自基類,并且這些類中包含虛函數,那么每個類將擁有各自獨立的虛函數表。下面我們來詳細解釋這個過程。

  1. 基類:在基類中,編譯器會為其創建一個虛函數表,這個表包含了基類中所有虛函數的地址。如果派生類沒有覆蓋(重寫)這些虛函數,派生類對象的虛函數表會復制基類虛函數表中相應的條目。

  2. 派生類:當派生類覆蓋(重寫)基類中的虛函數時,派生類的虛函數表中對應位置的函數指針會被更新為指向派生類中的函數實現。如果派生類引入了新的虛函數,這些新的虛函數也會被加入到派生類的虛函數表中。

  3. 多重繼承:在多重繼承的情況下,每個基類都會有自己的虛函數表。派生類對象會包含多個虛函數表指針,每個指針指向對應基類的虛函數表。如果派生類覆蓋了某個基類的虛函數,那么相關基類虛函數表中的條目會被更新為指向派生類中的實現。

示例理解

考慮以下示例:

class Base {
public:virtual void func1() { /* 實現 */ }virtual void func2() { /* 實現 */ }
};class Derived : public Base {
public:void func1() override { /* 新實現 */ }virtual void func3() { /* 新虛函數 */ }
};
  • Base 類有自己的虛函數表,包含 func1func2
  • Derived 類有自己的虛函數表,其中 func1 的條目會被更新為指向 Derived::func1func2 保持不變(因為它沒有被Derived重寫),并且會添加一個新的條目指向 func3

二、純虛函數(Pure Virtual Function)

2.1 定義和作用

純虛函數是在基類中聲明但不實現的虛函數,其聲明方式是在函數聲明的結尾處添加 = 0。類中如果包含至少一個純虛函數,則該類成為抽象類(Abstract Class),不能實例化對象。

純虛函數的主要作用是定義接口規范,強制要求派生類必須實現這些函數,從而實現接口的統一和標準化。

2.2 示例代碼

#include <iostream>
using namespace std;class Shape {
public:virtual void draw() = 0; // 純虛函數
};class Circle : public Shape {
public:void draw() override {cout << "Drawing a circle" << endl;}
};int main() {Shape* shape = new Circle();shape->draw(); // 輸出:Drawing a circledelete shape;return 0;
}

三、總結

虛函數和純虛函數是C++實現多態性的關鍵機制。通過虛函數,可以實現基類指針或引用調用派生類的成員函數;而純虛函數則定義了一個接口規范,使得派生類必須實現特定的函數。這兩種機制在C++面向對象編程中發揮著至關重要的作用。

理解虛函數和純虛函數的工作原理及其在C++中的應用,對于深入學習和掌握面向對象編程具有重要意義。希望本文能夠幫助讀者更好地理解這一概念,提升C++編程能力。

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

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

相關文章

2、事件機制、DOM操作、jquery對尺寸操作、jquery添加和刪除

一、事件機制 1、事件源.事件類型(事件處理程序) $(this)中的this不能加引號 $(#box).click(function () {$(this).css(background-color,blue)//點擊顏色變為藍色 })2、事件源.on/bind(事件類型&#xff0c;事件處理程序) $("#box").on(dbclick,function () {$(…

適配器模式在微服務的巧妙應用

適配器模式&#xff08;Adapter Pattern&#xff09;是一種結構型設計模式&#xff0c;它允許不兼容的接口之間可以一起工作。適配器模式通常用于將一個類的接口轉換成客戶端期望的另一種接口&#xff0c;從而使原本因接口不兼容而不能一起工作的類可以一起工作。 適配器模式的…

使用Haproxy搭建Web群集

Hapraxy是目前比較流行的一種群集調度工具&#xff0c;同類群集調度工具有很多&#xff0c;如LVS 和Nginx。相 比較而言&#xff0c;LVS.性能最好&#xff0c;但是搭建相對復雜:Nginx的 upstream 模塊支持群集功能&#xff0c;但是對群集節 點健康檢查功能不強&#xff0c;性能…

基于YOLOv8/YOLOv7/YOLOv6/YOLOv5的番茄成熟度檢測系統(Python+PySide6界面+訓練代碼)

摘要&#xff1a;開發番茄成熟度檢測系統對于提高農業產量和食品加工效率具有重大意義。本篇博客詳細介紹了如何利用深度學習構建一個番茄成熟度檢測系統&#xff0c;并提供了完整的實現代碼。該系統基于強大的YOLOv8算法&#xff0c;并結合了YOLOv7、YOLOv6、YOLOv5的對比&…

騰訊云幻獸帕魯服務器使用Linux和Windows操作系統,對用戶的技術要求有何不同?

騰訊云幻獸帕魯服務器使用Linux和Windows操作系統對用戶的技術要求有何不同&#xff1f; 首先&#xff0c;從操作界面的角度來看&#xff0c;Windows操作系統相對簡單易操作&#xff0c;適合那些偏好使用圖形化界面操作的用戶。而Linux操作系統則需要通過命令行完成&#xff0…

百度搜索引擎SEO優化方法

隨著互聯網的不斷發展&#xff0c;搜索引擎已經成為人們獲取信息、產品和服務的主要途徑之一。而在中國&#xff0c;百度作為最大的搜索引擎&#xff0c;其影響力不可忽視。了解并掌握百度SEO關鍵詞優化方法&#xff0c;對于提升網站在搜索引擎中的排名至關重要。 關鍵詞選擇&a…

數據結構——跳表

簡單介紹跳表 跳表&#xff08;Skip List&#xff09;是一種可以進行對數級別查找的數據結構&#xff0c;它通過在數據中構建多級索引來提高查詢效率。跳表是一種基于鏈表的隨機化數據結構&#xff0c;其本質是由多個鏈表組成&#xff0c;每個鏈表中的元素都是原始鏈表中的元素…

圖論 - Trie樹(字符串統計、最大異或對)

文章目錄 前言Part 1&#xff1a;Trie字符串統計1.題目描述輸入格式輸出格式數據范圍輸入樣例輸出樣例 2.算法 Part 2&#xff1a;最大異或對1.題目描述輸入格式輸出格式數據范圍輸入樣例輸出樣例 2.算法 前言 本篇博客將介紹Trie樹的常見應用&#xff0c;包括&#xff1a;Trie…

C++ 使用 nlohmann::json存儲json文件

C 使用 nlohmann::json存儲json文件 nlohmann::json 概述JSON 存儲的示例以追加的方式存儲json文件 nlohmann::json 概述 nlohmann::json 是 C 中一個流行的 JSON 庫&#xff0c;由 Niels Lohmann 開發。它提供了一個簡單而強大的 API&#xff0c;用于解析、構建、操作和序列化…

電子電氣架構——車載以太網協議棧

電子電氣架構——車載以太網協議棧 我是穿拖鞋的漢子&#xff0c;魔都中堅持長期主義的汽車電子工程師。 老規矩&#xff0c;分享一段喜歡的文字&#xff0c;避免自己成為高知識低文化的工程師&#xff1a; 沒有人關注你。也無需有人關注你。你必須承認自己的價值&#xff0c…

MySQL入門------數據庫與SQL概述

目錄 前言 一、數據庫相關概念 二、數據模型 1.關系型數據庫&#xff08;RDBMS&#xff09; 三、MySQL數據庫 1.下載和安裝 2.配置環境變量 四、SQL 1.SQL通用語法 2.SQL分類 前言 從本期開始&#xff0c;我們開始學習數據庫的相關理論和實踐知識&#xff0c;從入門…

jupyter 用pyecharts進行數據分析

一、jupyter和pyecharts下載和打開 因為我是用的pycharm&#xff0c;所以我直接在pycharm項目終端中下載pip install jupyter,pip install pyecharts 在你下載的項目路徑中輸入jupyter notebook 之后會進入頁面 Jupyter 具體使用參考這個鏈接&#xff1a;Jupyter Notebook基本…

Pygame教程01:初識pygame游戲模塊

Pygame是一個用于創建基本的2D游戲和圖形應用程序。它提供了一套豐富的工具&#xff0c;讓開發者能夠輕松地創建游戲和其他圖形應用程序。Pygame 支持許多功能&#xff0c;包括圖像和聲音處理、事件處理、碰撞檢測、字體渲染等。 Pygame 是在 SDL&#xff08;Simple DirectMed…

常用設計模式詳解

設計模式 1.UML圖 統一建模語言是用來設計軟件的可視化建模語言。定義了用例圖、類圖、對象圖、狀態圖、活動圖、時序圖、協作圖、構件圖、部署圖等 9 種圖。 1.1類圖 1.1.1類的表示方式 在UML類圖中&#xff0c;類使用包含類名、屬性(field) 和方法(method) 且帶有分割線…

基本正則表達式

基本正則表達式 正則命令功能&#xff3e;尖角號&#xff0c;用于模式的最左側&#xff0c;如“^oldbpy"&#xff0c;匹配以oldboy單詞開頭的行$美元符&#xff0c;用于模式的最右側&#xff0c;如"oldboy$"&#xff0c;表示以oldboy單詞結尾的行^$組合符&…

Java基于springboot的廚藝交流平臺的設計與實現代碼

摘 要 使用舊方法對廚藝交流信息進行系統化管理已經不再讓人們信賴了&#xff0c;把現在的網絡信息技術運用在廚藝交流信息的管理上面可以解決許多信息管理上面的難題&#xff0c;比如處理數據時間很長&#xff0c;數據存在錯誤不能及時糾正等問題。 這次開發的廚藝交流平臺功…

如何優雅的刪除undo表空間

前言 因磁盤空間不足&#xff0c;需要將undo表空間遷移到其它的存儲空間 本文介紹如何優雅的刪除undo表空間&#xff0c;并在新的存儲空間中創建新的undo表空間 詳細操作步驟如下&#xff1a; 1、查看默認undo表空間 SQL>show parameter undo NAME …

Redis的主從搭建

1.準備兩臺機器&#xff0c;安裝好redis 2.修改從服務器的redis配置 slaveof <masterip> <masterport>兩個參數 masterip 主的ip 主的端口號 masterport 3. 啟動redis 1.先啟動主機redis 2.再啟用從機redis 主機redis日志打印 從機redis 日志打印

【python】1.python3.12.2和pycharm社區版的安裝指南

歡迎來CILMY23的博客喔&#xff0c;本篇為【python】1.python3.12.2和pycharm社區版的安裝指南&#xff0c;感謝觀看&#xff0c;支持的可以給個一鍵三連&#xff0c;點贊關注收藏。 目錄 一、python3.12.2的下載與安裝 1.1下載 1.2安裝 二、pycharm的安裝 2.1下載安裝 2…

Bootstrap的使用

目錄 js的引入&#xff1a; 1.行內式 2.嵌入式 3.外鏈式 Bootstrap:的引入 注意事項&#xff1a; 條件注釋語句&#xff1a; 柵格系統&#xff1a; 列嵌套&#xff1a; 列偏移&#xff1a; 列排序&#xff1a; 響應式工具&#xff1a; Bootstrap的字體圖標的使用&a…