【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

相关文章

Pandas中统计汇总可视化函数plot()的使用

《Pandas中统计汇总可视化函数plot()的使用》Pandas提供了许多强大的数据处理和分析功能,其中plot()函数就是其可视化功能的一个重要组成部分,本文主要介绍了Pandas中统计汇总可视化... 目录一、plot()函数简介二、plot()函数的基本用法三、plot()函数的参数详解四、使用pl

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

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