哈喽大家好。
最近开始学习计算机基础相关的面试内容,比如C++、算法与数据结构、计算机网络、操作系统、设计模式、数据库等。
因此把学习时记的笔记分享给大家,希望对大家有所帮助。
今天分享的是C++中虚函数和虚函数表相关知识,下面是正文。
虚函数是在编译时,并不能确定的类函数,而是在运行时确定的。
核心点:通过基类对象访问派生类实现的函数。
虚函数的例子,通常有三步。
virtual
的。virtual
函数。virtual
函数,此时虽然是基类指针,但调用的是派生类实现的基类virtual
函数。
// 例子来源于: 菜鸟教程
class A
{
public:
virtual void foo()
{
cout<<"A::foo() is called"<<endl;
}
};
class B:public A
{
public:
void foo()
{
cout<<"B::foo() is called"<<endl;
}
};
int main(void)
{
A *a = new B();
a->foo(); // 在这里,a虽然是指向A的指针,但是被调用的函数(foo)却是B的!
return 0;
}
纯虚函数与虚函数的区别在于,纯虚函数的基类中的virtual
函数,只定义了,但不实现。实现交给派生类来做。
PS:带纯虚函数的类也叫抽象类,因为这种基类不能直接生成对象。
在基类中纯虚函数的方法的后面加 =0:
// 例子来源于: 菜鸟教程
virtual void funtion()=0
1、在动态分配堆上内存的时候,析构函数必须是虚函数
原因:动态分配堆上内存,无法自动回收。若基类指针指向派生类,然后基类指针调用delete
方法,只能释放基类的内存,无法释放派生类特有的部分内存,进而导致内存泄露。
析构函数定义成虚函数,基类指针调用delete
方法,会先调用派生类的析构函数,然后自动调用基类的析构函数。
析构函数必须是定义虚函数,但没有必要是纯虚的。
2、友元不支持虚拟函数
因为友元函数不是成员函数,只有成员函数才可以是虚函数。
另一个方面,虚函数的目的是通过基类对象访问派生类实现的函数,友元函数不是不是成员函数,更无继承关系。
3、虚函数必须要在基类实现,不实现,编译会报错
规定。
纯虚函数必须不能实现。
4、虚函数是C++实现多态的机制
C++多态指的是调用成员函数时,会根据调用函数的对象的类型来执行不同的函数。
而我们在基类定义了虚函数,并在派生类实现了虚函数,通过基类对象指针却可以指向派生类的实现的成员函数。
因此虚函数是C++实现多态的机制。
5、为什么析构函数必须是虚函数?为什么C++默认的析构函数不是虚函数?
将可能被继承的基类的构造函数设置为虚函数,可以防止用基类指针指向子类是,释放基类指针是可以释放掉子类独有的空间,进而防止内存泄漏。
6、静态函数和虚函数的区别
静态函数在编译期就确定了运行,而虚函数在运行期动态绑定,动态绑定的依据是虚函数指针和虚函数表。
因为额为增加虚函数指针和虚函数表,所以会带来额外的内存开销。
7、多态和虚函数
多态分为静态多态和动态多态。
虚函数表是由有虚函数的类生成的,简称为 V-Table
。
虚函数表由编译器生成,如果一个类有虚函数,那么该类就会生成一个4个字节的虚函数表指针,指向虚函数表。指针存储在对象实例的最前面位置。
虚函数表就可以理解为一个数组,每个单元用来存放虚函数的地址(下面有例子)。
下图展示了一个派生类继承基类虚函数,且没有重写基类的虚函数,对应的虚函数指针、虚函数表、和虚函数表对应指针调用的方法。可以看出:
下图展示了一个派生类重写基类虚函数,且重写基类的部分虚函数,对应的虚函数指针、虚函数表、和虚函数表对应指针调用的方法。可以看出:
多继承下的虚函数表,还是只有一个虚函数表。
多个基类之间的虚函数,按照继承的顺序,存放虚函数指针。
基类内部的虚函数,按照虚函数内部声明的顺序存放。
下图展示了多继承下的虚函数表,派生类直接继承基类虚函数,类似与单继承下的虚函数表的情况。
区别在于:
下图展示了多继承下的虚函数表,派生类重写部分基类虚函数,类似与单继承下的虚函数表的情况。
区别在于:
以上内容如有错误,欢迎指正,也欢迎大家一起交流。
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8