【C++八股题整理】虚函数

2024-08-26 12:52
文章标签 c++ 函数 整理 八股

本文主要是介绍【C++八股题整理】虚函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C++八股题整理 - 虚函数

  • 虚函数
    • 虚函数的定义?
    • C++11引入的override和final关键字的作用?
    • 虚函数的实现原理?虚函数表(vbtl)和虚函数表指针(vptr)
    • 虚函数表、虚函数表指针的生成时期及存储位置?
    • 含有虚函数的类的对象的大小?
    • 构造函数和析构函数可以是虚函数吗?
    • 构造函数和析构函数中能否调用虚函数?
    • 哪些函数不能是虚函数?
    • 虚函数和纯虚函数的区别?
    • 虚函数和模板的区别?

虚函数

虚函数的定义?

虚函数是在基类中使用关键字 virtual 声明的成员函数,它允许派生类对其进行重写(Override),实现运行时多态。当通过基类指针或引用调用虚函数时,实际调用的是对象类型对应的派生类中的函数,这个过程称为动态绑定(Dynamic Binding)

#include<iostream>  
using namespace std;  class A {  
public:  void foo() {  printf("1\n"); }  virtual void fun() {  printf("2\n");  }  // 虚函数
};  
class B : public A {  
public:  void foo() {  printf("3\n"); }  // 派生类的函数屏蔽了与其同名的基类函数void fun() {  printf("4\n"); }  // 重写虚函数
};  
int main(void) {  A a;  B b;  A *p = &a;  p->foo();  // 1p->fun();  // 2p = &b;  p->foo();  // 取决于指针类型,输出1p->fun();  // 取决于对象类型,输出4,体现了多态return 0;  
}

派生类B重写了A中的虚函数foo(),B中重写后的foo()同样是一个虚函数(不需要virtual显式标注)。如果B被继承,可以在子类中继续重写。

C++11引入的override和final关键字的作用?

  • override:保证在派生类中声明的重载函数,与基类的虚函数有相同的签名
    • 函数签名不一致:不加override,会视为派生类中新定义的函数;加了override,会报错
    virtual void fun() override;
    
  • final:阻止类的进一步派生 和 虚函数的进一步重写
    • 一个虚函数被定义为final,则派生类中不能再重写它
    virtual void fun() final;
    

虚函数的实现原理?虚函数表(vbtl)和虚函数表指针(vptr)

  • 类 的 虚函数表(vbtl)
    • 当一个类中包含虚函数时,编译器会为该类生成虚函数表,表中保存着该类包含的虚函数的地址。“包含”的意思是继承的+自己新定义的
    • 如果在该类中重写了父类的虚函数A,那就在虚函数表中将A对应的地方,替换成重写后的虚函数的地址
    • 类自己新定义的虚函数,也要将其追加某一张虚函数表上
    • 一个包含虚函数的类,至少有1张虚函数表,即使该类不重写任何虚函数
    • 一个类继承了n个有虚函数的基类,就有n张虚函数表
  • 对象 的 虚函数表指针(vptr)
    • 当一个类中包含虚函数时,该类的对象将会拥有虚函数表指针(vptr)指向该类的虚函数表。虚函数表指针也称虚指针、虚表指针
    • 类有n张虚函数表,类的对象就有n个虚指针,每个指针指向1张虚函数表
      在这里插入图片描述
  • 虚函数的实现原理
    在程序运行时,找到动态绑定到基类指针上的对象,然后根据该对象的虚函数表指针找到对应的虚函数表,从而确定调用哪个版本的虚函数。
    在这里插入图片描述

虚函数表、虚函数表指针的生成时期及存储位置?

  • 虚函数表:在编译时生成,存储在只读数据段
  • 虚函数表指针:在对象创建时生成,位置在对象的头部,根据对象创建方式存储在堆或栈上

含有虚函数的类的对象的大小?

前置知识:C++类对象大小的计算(一)常规类大小计算
含有虚函数的类的对象的大小 = 虚函数表指针(vptr)个数 x 指针大小 + 内存对齐后,对象拥有的非静态成员变量的大小

32位系统下,指针大小为4;64位系统下,指针大小为8。
在64位系统下考虑如下代码:

class Base1 {
public:int a;			// size: 4, 内存对齐后为 8static int b;	// 静态成员属于类,不计入大小virtual void func1() {}
};class Base2 {
public:double c;		// size: 8virtual void func2() {}
};class Derived : public Base1, public Base2 {
public:char d;			// size: 1, 内存对齐后为8virtual void func3() {}
};

Derived类的对象,共拥有a、c、d三个非静态成员变量,内存对齐后的总大小为8+8+8=24;Derived类的对象还拥有2个虚函数表指针,每个指针的大小为8,因此总的大小为24 + 2 x 8 = 40字节。

构造函数和析构函数可以是虚函数吗?

  • 构造函数不能是虚函数
    • vptr是在构造函数中初始化的,如果将构造函数定义为虚函数,那么在调用构造函数前vptr还未生成,因此无法调用到该构造函数
  • 析构函数应该为虚函数
    • 当基类的指针指向子类的对象时,如果基类的析构函数不为虚函数,那么销毁基类指针时,只会调用基类的析构函数,子类的对象无法被析构,造成内存泄漏

构造函数和析构函数中能否调用虚函数?

在构造函数和析构函数中调用虚函数,不会起到想要的结果。比较下面两段代码:

#include <iostream>class Base {
public:virtual void show() {std::cout << "Base show()" << std::endl;}void callShow() {std::cout << "Base callShow()" << std::endl;show();  // 调用虚函数}virtual ~Base() = default;
};class Derived : public Base {
public:void show() override {std::cout << "Derived show()" << std::endl;}
};int main() {Derived d;d.callShow();  // 调用基类的成员函数,但期望调用派生类的虚函数return 0;
}
// Base callShow()
// Derived show()

这段代码中,虚函数show正确地表现出了多态性。而在构造函数和析构函数中调用,不能表现多态性。

#include <iostream>class Base {
public:Base() {std::cout << "Base constructor" << std::endl;show();  // 调用虚函数}virtual void show() {std::cout << "Base show()" << std::endl;}virtual ~Base() {std::cout << "Base destructor" << std::endl;show();  // 再次调用虚函数}
};class Derived : public Base {
public:Derived() {std::cout << "Derived constructor" << std::endl;}void show() override {std::cout << "Derived show()" << std::endl;}~Derived() {std::cout << "Derived destructor" << std::endl;}
};int main() {Derived d;return 0;
}
// Base constructor
// Base show()
// Derived constructor
// Derived destructor
// Base destructor
// Base show()

在构造函数和析构函数中调用show(),show采用的是基类中的实现。这是因为,在调用到基类的构造函数和析构函数时,派生类中的内容尚没有被创建、或者已经被销毁了。

哪些函数不能是虚函数?

  • 构造函数:执行构造函数前虚表指针尚未初始化,无法正确调用构造函数
  • 内联函数:内联函数在编译阶段进行函数体的替换操作,而虚函数意味着在运行期间进行类型确定,所以内联函数不能是虚函数
  • 静态函数:静态函数不属于对象属于类,静态成员函数没有this指针,因此静态函数设置为虚函数没有任何意义
  • 友元函数,友元函数不属于类的成员函数,不能被继承。对于没有继承特性的函数没有虚函数的说法
  • 普通函数,普通函数不属于类的成员函数,不具有继承特性,因此普通函数没有虚函数

总结:不能被继承的函数 和 不能被重写的函数 不能是虚函数

虚函数和纯虚函数的区别?

class A {  virtual void example() = 0;	// 纯虚函数
}

纯虚函数是虚函数的一种特殊形式,它的语法是在函数声明后加上’=0’。纯虚函数只有声明没有实现,含有纯虚函数的类称为抽象类,不能被实例化。它的派生类如果想被实例化,就必须实现所有的纯虚函数。

  • 虚函数和纯虚函数都是实现多态性的工具。通过将基类的指针或引用指向派生类对象,可以在运行时调用派生类的重写方法。
  • 虚函数提供了一个默认实现,但派生类可以选择重写它。纯虚函数则强制要求派生类必须提供自己的实现。

虚函数和模板的区别?

模版是一种编译时多态性技术,通过在编译时确定类型来生成特定的代码。
虚函数是运行时多态,在运行时根据对象的实际类型来调用相应的方法,从而实现多态性。

特性模板(Templates)虚函数(Virtual Functions)
决策时间编译时运行时
实现机制编译时生成特定类型代码通过虚函数表动态绑定
类型检查编译时运行时
运行时开销有虚函数表查找开销
使用场景泛型编程,STL容器面向对象编程的多态行为

这篇关于【C++八股题整理】虚函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1108621

相关文章

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

MySQL高级查询之JOIN、子查询、窗口函数实际案例

《MySQL高级查询之JOIN、子查询、窗口函数实际案例》:本文主要介绍MySQL高级查询之JOIN、子查询、窗口函数实际案例的相关资料,JOIN用于多表关联查询,子查询用于数据筛选和过滤,窗口函... 目录前言1. JOIN(连接查询)1.1 内连接(INNER JOIN)1.2 左连接(LEFT JOI

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

MySQL中FIND_IN_SET函数与INSTR函数用法解析

《MySQL中FIND_IN_SET函数与INSTR函数用法解析》:本文主要介绍MySQL中FIND_IN_SET函数与INSTR函数用法解析,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一... 目录一、功能定义与语法1、FIND_IN_SET函数2、INSTR函数二、本质区别对比三、实际场景案例分

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序