刨析一下C++构造析构函数能不能声明为虚函数的背后机理?
以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「englyf」https://www.cnblogs.com/englyf/p/16631774.html
先说结论:
构造函数不能声明为虚函数,析构函数可以声明为虚函数。
构造函数可以声明为虚函数吗?
虚函数表里都存了些什么东西?不是金,不是银,是对应类里声明为虚函数的成员地址。在编译期,每个类的虚函数表即被分配和生成。同一个类的所有实例对象都是共享这个虚函数表的,那么每个实例对象也就会隐含有一个成员指针变量专门用来存储虚函数表的地址。这个隐含的成员指针变量需要在实例对象初始化后才会指向虚函数表。
很显然,对象不能在没有初始化之前就知道自己对应的虚函数表在哪里,因此也不能在对象初始化之前调用访问虚函数表的内容(虚函数)。是不是在对象初始化之前就不能调用访问这些虚函数成员了?不是这个意思,这里说的不能仅限于通过虚函数表来访问(比如派生后的类实例对象通过父类的指针变量访问调用虚函数成员),也就是动态访问时才需要虚函数表的信息。不过,就算某个成员函数被声明为虚函数时,也可以通过类的静态特性合法访问的,这是无关痛痒的题外话,评价____。
对象的初始化就是依赖于构造函数的执行,首先是找到最上一层父类的构造函数并执行,然后逐层往下执行构造函数,直到执行完当前类(被实例化的类)的构造函数,这个过程不需要虚函数表的任何信息,在编译期就确定了所有需要的信息了。在构造函数执行之前,对象还无法确定自身的虚函数表在哪里,又怎么从虚函数表里查找对应的虚构造函数呢?如果把构造函数声明为虚函数,那么意义在哪儿呢?想来想去,可能费劲打造出一把刀刃能削铁如泥的好刀,却只是在需要敲钉子时,想起用这把刀的刀背。
所以,很明显构造函数不能声明为虚函数。
析构函数可以声明为虚函数吗?
先看下面的代码,析构函数不声明为虚函数时,
#include <iostream>
using namespace std;
class Base
{
public:
Base() {
cout << "Base constructor" << endl;
}
~Base() {
cout << "Base destructor" << endl;
}
};
class Derived : public Base
{
public:
Derived() {
cout << "Derived constructor" << endl;
}
~Derived() {
cout << "Derived destructor" << endl;
}
};
class Derived_again : public Derived
{
public:
Derived_again() {
cout << "Derived_again constructor" << endl;
}
~Derived_again() {
cout << "Derived_again destructor" << endl;
}
};
int main()
{
Base *obj = new Derived();
cout << "----" << endl;
delete obj;
return 0;
}
看看编译后执行的结果(这里用的编译器是g++)
Base constructor
Derived constructor
----
Base destructor
从上面的输出来看,类Derived的析构函数没有被调用到,这会导致典型的问题--内存泄漏。
然后把所有析构函数声明为虚函数,重新编译再看看执行结果
Base constructor
Derived constructor
----
Derived destructor
Base destructor
可以看到,需要调用的析构函数都调用了。
那么怎么去理解上面这段代码的执行逻辑呢?
delete obj;
销毁对象时,系统执行的逻辑有两种情况。
一种是,如果析构函数没有被声明为虚函数时,那么在指针obj指向的对象的虚函数表里,是找不到虚析构函数的。由于系统无法往下查找派生类的内容,而且变量obj被声明为某个类(上面代码对应的是Base)的指针类型,那么系统就转为调用这个类(Base)的成员析构函数,这里利用的是静态特性。
另一种是,如果析构函数被声明为虚函数时,先在指针obj指向的对象的虚函数表里,尝试寻找当前对象obj的虚析构函数,找到后执行它。对应代码,被实例化的类是Derived,对象obj的虚析构函数地址应该指向类Derived的成员析构函数。
上面两种情况中,找到第一个析构函数并执行后,会按照静态特性(也就是按照编译期生成的信息),逐层往上一层父类查找成员析构函数并执行,直到最上一层的父类的析构函数被执行完毕为止。
于是,第一种情况下,类Derived的成员析构函数被漏掉执行了,这会导致类Derived所申请的资源没有对应的析构函数来执行释放,内存泄漏发生地那么顺理成章。
可以看到,动态特性可以把事情玩得妥妥当当,面向对象的高级感可能就来自这里。
总结一下,很明显,析构函数可以声明为虚函数,但不是必须。某些情况下,也是必须的,比如,当类指针指向的是该类的子类实例时,析构函数必须声明为虚函数,以防止内存泄漏。
刨析一下C++构造析构函数能不能声明为虚函数的背后机理?的更多相关文章
- (C++)浅谈多态基类析构函数声明为虚函数
主要内容: 1.C++类继承中的构造函数和析构函数 2.C++多态性中的静态绑定和动态绑定 3.C++多态性中析构函数声明为虚函数 1.C++类继承中的构造函数和析构函数 在C++的类继承中, 建立对 ...
- 从零开始学C++之虚函数与多态(一):虚函数表指针、虚析构函数、object slicing与虚函数
一.多态 多态性是面向对象程序设计的重要特征之一. 多态性是指发出同样的消息被不同类型的对象接收时有可能导致完全不同的行为. 多态的实现: 函数重载 运算符重载 模板 虚函数 (1).静态绑定与动态绑 ...
- C++多态性中基类析构函数声明为虚函数
在用基类指针指向派生类时, 在基类析构函数声明为virtual的时候,delete基类指针,会先调用派生类的析构函数,再调用基类的析构函数. 在基类析构函数没有声明为virtual的时候,delete ...
- C++ 构造函数和析构函数的调用顺序、虚析构函数的作用
构造函数和析构函数的调用顺序 构造函数的调用顺序: 当建立一个对象时,首先调用基类的构造函数,然后调用下一个派生类的构造函数,依次类推,直至到达最底层的目标派生类的构造函数为止. 析构函数的调用书序: ...
- C++ 析构函数为虚函数
1.原因: 在实现多态时, 当用基类指针操作派生类, 在析构时候防止只析构基类而不析构派生类. 2.例子: (1). #include<iostream> using namespace ...
- C++箴言:避免构造或析构函数中调用虚函数
如果你已经从另外一种语言如C#或者Java转向了C++,你会觉得,避免在类的构造函数或者析构函数中调用虚函数这一原则有点违背直觉.但是在C++中,违反这个原则会给你带来难以预料的后果和无尽的烦恼. 正 ...
- C++中的new/delete、构造/析构函数、dynamic_cast分析
1,new 关键字和 malloc 函数区别(自己.功能.应用): 1,new 关键字是 C++ 的一部分: 1,如果是 C++ 编译器,则肯定可以用 new 申请堆空间内存: 2,malloc 是由 ...
- C++中为什么要将析构函数定义成虚函数
构造函数不可以是虚函数的,这个很显然,毕竟虚函数都对应一个虚函数表,虚函数表是存在对象内存空间的,如果构造函数是虚的,就需要一个虚函数表来调用,但是类还没实例化没有内存空间就没有虚函数表,这根本就是个 ...
- C++析构函数定义为虚函数(转载)
转载:http://blog.csdn.net/alane1986/article/details/6902233 析构函数执行时先调用派生类的析构函数,其次才调用基类的析构函数.如果析构函数不是虚函 ...
随机推荐
- Python收集这些视频只是单纯的想做做壁纸,大家不要误会
首先澄清一下,我用Python收集这些视频,绝不是想做别的什么,真的只是用来做动态壁纸,大家不要误会!我不是那样的人~ 这样的不过份吧 (这个动图看不看的到就看有没有缘分了 ) 阅读本文你需要准备 1 ...
- JAVA中简单的for循环竟有这么多坑,你踩过吗
JAVA中简单的for循环竟有这么多坑,你踩过吗 实际的业务项目开发中,大家应该对从给定的list中剔除不满足条件的元素这个操作不陌生吧? 很多同学可以立刻想出很多种实现的方式,但你想到的这些实现方式 ...
- PTA(BasicLevel)-1023 组个最小数
一. 问题定义 给定数字 0-9 各若干个.你可以以任意顺序排列这些数字,但必须全部使用.目标是使得最后得到的数尽可能小(注意 0 不能做首位). 例如:给定两个 0,两个 1,三个 5,一个 8,我 ...
- 2019 CSP-J 初赛解析
题面,成绩不是真实水平,就挑重点说一说 老师给的解析 T5 这是二分查找,属于是我的代码理解不太对 我的理解 #include<iostream> using namespace std; ...
- Ubuntu修改网卡名
vim /etc/udev/rules.d/70-persistent-net.rules 添加以下内容: SUBSYSTEM=="net", ACTION=="add& ...
- 基于 Rainbond 部署 DolphinScheduler 高可用集群
本文描述通过 Rainbond 云原生应用管理平台 一键部署高可用的 DolphinScheduler 集群,这种方式适合给不太了解 Kubernetes.容器化等复杂技术的用户使用,降低了在 Kub ...
- Webpack干货系列 | 怎么运用 Webpack 5 处理css/scss/sass、less、stylus样式资源
程序员优雅哥简介:十年程序员,呆过央企外企私企,做过前端后端架构.分享vue.Java等前后端技术和架构. 本文摘要:主要讲解webpack 5 如何高效处理CSS 资源.scss/sass 资源.l ...
- ACWing93.递归实现组合型枚举
题面 \93. 递归实现组合型枚举 从 1∼n 这 n 个整数中随机选出 m 个,输出所有可能的选择方案. 输入格式 两个整数 n,m ,在同一行用空格隔开. 输出格式 按照从小到大的顺序输出所有方案 ...
- CF1700C Helping the Nature
题目大意: 给出一个长度为 n 的序列 a,每次可以进行三种操作中的一种: 选择i,将 a_1,a_2,...,a_i减1. 选择i,将 a_i,a_i+1,...,a_n减1. 将所有 a_i加1. ...
- 2022-7-13 第五组 pan小堂 java基础
###java基础 1.java语言发展史和概述平台(了解) 詹姆斯·高斯林(James Gosling)1977年获得了加拿大卡尔加里大学计算机科学学士学位,1983年获得了美国卡内基梅隆大学计算机 ...