EC笔记,第二部分:9.不在构造、析构函数中调用虚函数

9.不在构造、析构函数中调用虚函数

1.在构造函数和析构函数中调用虚函数会产生什么结果呢?

#include <iostream>

using namespace std;

class cls1{

public:

cls1(){

newMake();

};

~cls1(){

deleteIt();

};

virtual void newMake(){

cout<<"cls1 make"<<endl;

}

virtual void deleteIt(){

cout<<"cls1 delete"<<endl;

}

};

class cls2:public cls1

{

public:

cls2(){

newMake();

};

~cls2(){

deleteIt();

};

virtual void newMake(){

cout<<"cls2 make"<<endl;

}

virtual void deleteIt(){

cout<<"cls2 delete"<<endl;

}

};

int main(int argc, char const *argv[])

{

cls2 c;

return 0;

}

上述程序会产生什么样的输出呢?

你一定会以为会输出:

cls2 make

cls2 delete

或者是:

cls2 make

cls2 make

cls2 delete

cls2 delete

(如果你想到了后一种,说明你对派生类的构造有一定了解)

因为在构造和析构函数使用了虚函数,应该是迟绑定。但是实际的输出是:

cls1 make

cls2 make

cls2 delete

cls1 delete

为什么呢?

理由如下:

在程序构造一个cls2对象的时候,会调用基类(也就是cls1)的构造函数,但是这时cls2还没有被构造,所以虚表中只有cls1的newMake函数,所以程序只能调用cls1的newMake,然后到cls2的构造函数体执行,这时cls2已经被构造,所以cls2的构造函数会调用cls2的newMake

析构的时候,cls2的部分会率先释放,等到cls1释放的时候,虚表中已经不存在deleteIt的cls2版本了

2.解决方案

不在构造函数中调用虚函数,而是写一个通用的构造函数(或者通用的非构造函数,在基类构造函数中调用)

时间: 2024-10-21 11:02:11

EC笔记,第二部分:9.不在构造、析构函数中调用虚函数的相关文章

C++ 笔记(二) —— 不要在构造和析构函数中调用虚函数

ilocker:关注 Android 安全(新手) QQ: 2597294287 1 class Transaction { //所有交易的 base class 2 public: 3 Transaction(); 4 virtual void logTransaction() const = 0; //做出一份因类型不同而不同的日志记录 5 … 6 } 7 Transaction::Transaction() { 8 … 9 logTransaction(); 10 } derived cl

条款9:绝不要在构造以及析构函数中调用虚函数

在构造以及析构函数期间不要调用virtual函数,因为这类调用从不下降到derived class中.例如说下面这个例子: 1 class Transaction{ 2 public: 3 Transaction(); 4 virtual void logTransactions()s const = 0; 5 //... 6 }; 7 Transaction::Transaction() 8 { 9 //... 10 logTransaction(); 11 } 12 class BuyTra

第八章:不要在构造和析构函数中使用虚函数

前言 本文将讲解一个新手C++程序员经常会犯的错误 - 在构造/析构函数中使用虚函数,并分析错误原因所在以及规避方法. 错误起因 首先,假设我们以一个实现交易的类为父类,然后一个实现买的类,一个实现卖的类为其子类. 这三个类的对象初始化过程中,都需要完成注册的这么一件事情 (函数).然而,各自注册的具体行为是不同的. 有些人会写出以下这样的代码: 1 class Transaction { 2 public: 3 Transaction(); // 父类构造函数 4 //...... 5 pri

不要在构造和析构函数中使用虚函数

前言 本文将讲解一个新手 C++ 程序员经常会犯的错误 - 在构造/析构函数中使用虚函数,并分析错误原因所在以及规避方法. 错误起因 首先,假设我们以一个实现交易的类为父类,然后一个实现买的类,一个实现卖的类为其子类. 这三个类的对象初始化过程中,都需要完成注册的这么一件事情 (函数).然而,各自注册的具体行为是不同的. 有些人会写出以下这样的代码: 1 class Transaction { 2 public: 3 Transaction(); // 父类构造函数 4 //...... 5 p

Item 9:在析构/构造时不要调用虚函数 Effective C++笔记

Item 9: Never call virtual functions during construction or destruction. 父类构造期间,对虚函数的调用不会下降至子类.如果这并非你的意图,请不要这样做! 这个问题阿里实习面试曾经问到过,看这篇文章: 2014阿里巴巴面试经历 看Scott Meyers举的例子: class Transaction { // base class for all public: // transactions Transaction(){ /

绝不在构造和析构函数中调用 virtual 函数

看下面的这段代码,问 print调用的是基类还是派生类的版本? 答案是 基类... 可能大家会很惊讶,print不是virtual function 吗?为什么不是调用派生类的版本呢? 首先,当定义一个派生类的对象的时候, 由于 base class 构造函数的执行更早于 derived class构造函数, 所以当 base class constructor 调用的时候,派生类的成员尚未初始化(说明,这个时候真正的 虚函数表尚未完全初始化). 如果这个时候调用 派生类的函数(可能使用未初始化

Effective C++_笔记_条款09_绝不在构造和析构过程中调用virtual函数

(整理自Effctive C++,转载请注明.整理者:华科小涛@http://www.cnblogs.com/hust-ghtao/) 为方便采用书上的例子,先提出问题,在说解决方案. 1 问题 1: class Transaction{ 2: public: 3: Transaction(); 4: virtual void LogTransaction() const = 0 ; 5: ... 6: }; 7:  8: Transaction::Transaction() //Base cl

写了一个二叉树构造及中序遍历函数

本题就是测试读入数据的速度的. 如果有大量的数据读入,使用cin是很慢的. 那么使用scanf那么会快很多,但是如果数据量更大的话那么就还是不够快了. 所以这里使用fread. 首先开一个buffer,然后使用fread大块大块地读入数据就可以非常快地读入了. 题目如下: Input The input begins with two positive integers n k (n, k<=107). The next n lines of input contain one positive

Effective C++ 条款九、十 绝不在构造和析构过程中调用virtual函数|令operator=返回一个reference to *this

  1.当在一个子类当中调用构造函数,其父类构造函数肯定先被调用.如果此时父类构造函数中有一个virtual函数,子类当中也有,肯定执行父类当中的virtual函数,而此时子类当中的成员变量并未被初始化,所以无法调用子类与之对应的函数.即为指向虚函数表的指针vptr没被初始化又怎么去调用派生类的virtual函数呢?析构函数也相同,派生类先于基类被析构,又如何去找派生类相应的虚函数? 2.做法:将子类的某个函数改为non-virtual,然后在子类构造函数中传递参数给父类函数.然后父类的构造函数