C++中虚函数和纯虚函数的问题总结

简介: C++中虚函数和纯虚函数的问题总结



 

C++中虚函数和纯虚函数的问题总结

虚函数

虚函数的定义

在C++中,在成员函数的声明中加上关键字virtual即可声明该函数为虚函数。

虚函数的目的

虚函数在面向对象的程序设计中实现了多态中的动态绑定问题:

面向对象程序设计中,派生类继承自基类。使用基类指针引用访问派生类对象时。如果派生类覆盖了基类中的方法,通过上述指针或引用调用该方法时,可以有两种结果:

  1. 调用到基类的方法:编译器根据指针或引用的类型决定,称作“静态绑定”;
  2. 调用到派生类的方法:语言的运行时系统根据对象的实际类型决定,称作“动态绑定”。

虚函数的效果属于后者。

如果基类中的函数是“虚”的,则调用到的都是在运行时最终派生类中的函数实现,与指针或引用的类型无关。

如果函数非“虚”,调用到的函数就在编译期根据指针或者引用所指向的类型决定。

例如:

#include <iostream>
using namespace std;
class Base {
public:
    virtual void Max() {
        cout << "Max 1" << endl;
    }
};
class Son : public Base{//继承
public:
    void Max() {
        cout << "Max 2" << endl;
    }
    void Min() {
        cout << "Min" << endl;
    }
};
int main() {
    cout << "----------1------------" << endl;
    Base base;
    base.Max();
    cout << "----------2------------" << endl;
    Base* base1 = new Son;//指针访问
    base1->Max();
    //base1.Min();  //会报错,无法调用非虚函数
    cout << "----------3------------" << endl;
    Son son1;
    Base& base3 = son1; //引用访问实现动态绑定,创建时必须初始化
    base3.Max();
    //base3.Min();  //会报错,无法调用非虚函数
    return 0;
}

结果:

 

注意:引用访问实现动态绑定时,创建对象时必须初始化;指针则可以先声明再初始化。并且只能调用虚函数(调用非虚函数会报错,如例子中的Min()函数所示)。

继承时,虚函数的属性也被继承了,就算子类重写的成员函数中不写“virtual“,系统会默认它是虚函数。

例如:

#include <iostream>
using namespace std;
class Base {
public:
    virtual void Max() {
        cout << "Max 1" << endl;
    }
};
class Son : public Base{
public:
    void Max() {
        cout << "Max 2" << endl;
    }
};
class GrandSon : public Son{//
public:
    void Max() {
        cout << "Max 3" << endl;
    }
};
int main() {
    cout << "----------1------------" << endl;
    Base base;
    base.Max();
    cout << "----------2------------" << endl;
    Base* base1 = new GrandSon;
    base1->Max();
    return 0;
}

其中,Son类中的Max()函数没有标明“virtual”,但是它是重写的Base类中的函数,因此会默认为虚函数。因此在GrandSon类中,Max()函数仍能实现重写操作,能够实现动态绑定。(对重载、重写、隐藏有疑问的可以参考一文彻底解决C++中的重载、重写和隐藏操作

结果:

 

纯虚函数

纯虚函数的定义

纯虚函数只是相当于一个接口名,含有纯虚函数的类不能够实例化。纯虚函数首先是虚函数,其次它没有函数体,取而代之的是用“=0”。

纯虚函数的特点

一个类中如果有纯虚函数的话,称其为抽象类。抽象类不能用于实例化对象,否则会报错。抽象类一般用于定义一些公有的方法。子类继承抽象类也必须实现其中的纯虚函数才能实例化对象。

例如:

#include <iostream>
using namespace std;
class Base {
public:
    virtual void Max() {
        cout << "Max 1" << endl;
    }
    virtual void Min() = 0; //纯虚函数
};
class Son : public Base{//继承
public:
    void Max() {
        cout << "Max 2" << endl;
    }
    void Min() {
        cout << "Min" << endl;//子类实现纯虚函数
    }
};
int main() {
    cout << "----------1------------" << endl;
    //Base base; //报错,含有纯虚函数,不能实例化
    //base.Max();
    cout << "----------2------------" << endl;
    Base* base1 = new Son;//指针访问
    base1->Max();
    base1->Min();  //
    return 0;
}

结果:

 

虚函数相关问题

问题一:基类的虚函数表存放在内存的什么位置,虚表指针vptr的初始化时间

虚函数表vtable在Linux/Unix中存放在可执行文件的只读数据段中(rodata),也就是内存模型中的常量区中。而虚函数位于代码区

由于虚表指针vptr跟虚函数密不可分,对于有虚函数或者继承于拥有虚函数的基类,对该类进行实例化时,在构造函数执行时会对虚表指针进行初始化,并且存在对象内存布局的最前面。

问题二:虚函数能否声明为内联函数?

首先,声明为inline的函数,编译器也不一定真实实现内联,还得看函数简不简单。

将虚函数声明为inline,要分情况讨论。因为虚函数可以实现多态(在运行时决定),而inline是在编译时由编译器决定是否内联。所以将虚函数声明为inline分为以下两种情况:

1.当是指向派生类的指针(多态性)调用声明为inline的虚函数时,不会内联展开;

2.当是对象本身调用虚函数时,会内联展开(前提是函数并不复杂的情况下),但是这也并没有实现虚函数的作用。

问题三:构造函数为什么不能为虚函数?析构函数为什么要虚函数?

构造函数:

构造函数不能定义为虚函数。在构造函数中可以调用虚函数,不过此时调用的是正在构造的类中的虚函数,而不是子类的虚函数,因为此时子类尚未构造好。

原因:虚函数对应一个vtable(虚函数表),类中存储一个vptr指向这个vtable。如果构造函数是虚函数,就需要通过vtable调用,可是对象没有初始化就没有vptr,无法找到vtable,所以构造函数不能是虚函数。

析构函数:

析构函数可以为虚函数,并且一般情况下基类析构函数要定义为虚函数。

原因:从上面析构函数的调用顺序可知,首先要调用派生子类的析构函数。因此只有在基类析构函数定义为虚函数时,调用操作符delete销毁指向对象的基类指针时,才能准确调用派生类的析构函数(从该级向上按序调用虚函数),才能准确销毁数据。

另外析构函数可以是纯虚函数,含有纯虚函数的类是抽象类,此时不能被实例化。但派生类中可以根据自身需求重新改写基类中的纯虚函数。

问题四:构造函数和析构函数可以调用虚函数吗?

在C++中,可以调用,但不提倡;

① 我们调用虚函数,一般就是使用其动态联编的功能。但是构造函数和析构函数调用虚函数时都不使用动态联编,如果在构造函数或析构函数中调用虚函数,则运行的是为构造函数或析构函数自身类型定义的版本;

② 构造函数时,因为父类对象会在子类之前进行构造,此时子类部分的数据成员还未初始化,因此C++不会进行动态联编;

③析构函数是用来销毁一个对象的,在销毁一个对象时,先调用子类的析构函数,然后再调用基类的析构函数。所以在调用基类的析构函数时,派生类对象的数据成员已经销毁,也没有使用动态联编。

问题五:静态成员函数static能定义为虚函数吗?常成员函数const呢?

static静态成员函数不能定义为虚函数。

static成员不属于任何类对象或类实例,静态成员函数没有this指针。

虚函数依靠vptr和vtable来处理。vptr是一个指针,在类的构造函数中创建生成,并且只能用this指针来访问它,因为它是类的一个成员,并且vptr指向保存虚函数地址的vtable。而对于静态成员函数,它没有this指针,所以无法访问vptr。

const成员函数可以是虚函数。这样声明该函数都是“只读”操作。

问题六:哪些函数不能是虚函数?

① 构造函数

② 内联函数

③ 静态函数,静态函数不属于对象属于类,静态成员函数没有this指针,因此静态函数设置为虚函数没有任何意义。

④ 友元函数,友元函数不属于类的成员函数,不能被继承。对于没有继承特性的函数没有虚函数的说法。

⑤ 普通函数,普通函数不属于类的成员函数,不具有继承特性,因此普通函数没有虚函数。

目录
相关文章
|
7月前
|
C++
C++一分钟之-虚函数与抽象类
【6月更文挑战第21天】在C++中,虚函数与抽象类是多态的基础,增进类间耦合与灵活性。虚函数实现动态绑定,抽象类定义不可实例化的接口。关键点包括:记得使用`virtual`,避免滥用虚函数,确保派生类实现纯虚函数。抽象类不能直接实例化,派生类必须实现所有纯虚函数。通过实例代码学习和实践,能更好地掌握这些概念以优化代码设计。
62 2
|
12天前
|
Serverless 编译器 C++
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
33 4
|
12天前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
26 3
|
8月前
|
C++
C++程序中的纯虚函数
C++程序中的纯虚函数
65 0
|
5月前
|
编译器 C++ 索引
C++虚拟成员-虚函数
C++虚拟成员-虚函数
|
8月前
|
Serverless C++
C++多态性、虚函数、纯虚函数和抽象类知识网络构造
C++多态性、虚函数、纯虚函数和抽象类知识网络构造
|
8月前
|
C++ 开发者
C++程序中利用虚函数实现动态多态性
C++程序中利用虚函数实现动态多态性
78 2
|
8月前
|
C++ 编译器 存储
|
8月前
|
数据安全/隐私保护 C++
C++中的虚函数、纯虚函数与函数重写的技术性探讨
C++中的虚函数、纯虚函数与函数重写的技术性探讨
101 0
|
8月前
|
编译器 C++
C++的虚函数
C++的虚函数
47 0