绝不在构造和析构过程中调用虚函数

2024-02-13 20:08

本文主要是介绍绝不在构造和析构过程中调用虚函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、中心内容

因为类调用从不下降至派生类,导致若为纯虚函数,则找不到函数的实现代码;若为非虚函数,则可能会导致调用错误的函数版本。。。

二、内容简介

考虑这样一系列继承:

class Transaction{

public:

    Transaction();

    virtual void logTransaction() const = 0;//每创建一个交易对象,在审计日志中需要创建一笔适当记录,由此函数实现

    ......

};

Transaction::Transaction()

{

    ......

    logTranction();

}

class BuyTranction: public Transaction{

public:

    virtual void logTransaction() const;

    ......

};

class SellTranction : public Transaction{

public:

    virtual void logTransaction() const;

    ......

};

note:

执行语句:BuyTransaction b; 时发生的事情:

(1)派生对象内的基类成分会在派生自身成分被构造之前先构造妥当;

(2)红色语句被调用的是Transaction内的版本,不是BuyTransaction的版本!!!!!!

(3)即,在基类构造期间, 虚函数不是虚函数,不会下降到派生类阶层。

(4)或者说,派生类的基类部分构造期间,对象的类型是基类而不是派生类!!!!!!

具体原因:

因为基类构造函数执行时,派生类的成员变量尚未初始化,若下降到派生阶层,必然会用到局部成员变量,但是这些变量尚未初始化,会出现错误。


针对红色语句部分,基类构造函数调用了虚函数,这是不被允许的,具体原因:

1、纯虚函数

由于一般情况下,纯虚函数在基类中是不做定义的,所以调用的时候会找不到函数的定义代码,不管是基类构造函数还是派生类构造函数构造基类成分时;

2、非纯虚函数

原本构造一个派生类对象需要调用的是派生类版本的logTransaction(),但是由于基类的构造函数中调用了基类版本的

logTransaction(),所以构造基类成分时, 会调用该版本,造成错误版本的调用。

解决办法:

将基类的相应虚函数改为非虚函数,然后要求派生类构造函数传递必要信息给基类的构造函数,然后基类的构造函数就可以安全地调用非虚函数。具体实例如下:

class Transaction{

public:

    explicit Transaction(const std::string& logInfo);

    void logTransaction(const std::string& logInfo) const;//去掉了虚函数属性,使得整个继承中只有一个版本的此函数

    ......

};

Transaction::Transaction(const std::string& logInfo)

{

    ......

logTransaction(logInfo);

}

class BuyTransaction: public Transaction{

public:

    BuyTransaction(parameters)

    : Transaction(createLogString (parameters))

{......}

...

private:

    static std::string createLogString(parameters);

};

note:

1、红色部分就是派生类为基类构造函数提供的必要的logTransaction信息;

2、绿色部分的私有成员之所以定义为静态成员,在于静态成员函数不包含this指针,这样就不会指向其他尚未初始化的成员变量,避免了构造基类成分是出现错误。


note:

最重要的是!!!

因为使用非纯虚函数会使得在构造派生对象时,创建基类成分过程中,调用基类版本的虚函数导致错误,不能实现向下调用!!!

所以,在派生类中创建这么一个私有静态函数,增加可读性,因为是static又不会包含this指针指向派生类中未初始化的成员变量,向上传递因类而异的不同的构造信息给基类成分的构造函数,这样就不会调用错虚函数。。

这篇关于绝不在构造和析构过程中调用虚函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/706488

相关文章

oracle 11g导入\导出(expdp impdp)之导入过程

《oracle11g导入导出(expdpimpdp)之导入过程》导出需使用SEC.DMP格式,无分号;建立expdir目录(E:/exp)并确保存在;导入在cmd下执行,需sys用户权限;若需修... 目录准备文件导入(impdp)1、建立directory2、导入语句 3、更改密码总结上一个环节,我们讲了

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

Java Kafka消费者实现过程

《JavaKafka消费者实现过程》Kafka消费者通过KafkaConsumer类实现,核心机制包括偏移量管理、消费者组协调、批量拉取消息及多线程处理,手动提交offset确保数据可靠性,自动提交... 目录基础KafkaConsumer类分析关键代码与核心算法2.1 订阅与分区分配2.2 拉取消息2.3

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Python Counter 函数使用案例

《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

Java调用Python脚本实现HelloWorld的示例详解

《Java调用Python脚本实现HelloWorld的示例详解》作为程序员,我们经常会遇到需要在Java项目中调用Python脚本的场景,下面我们来看看如何从基础到进阶,一步步实现Java与Pyth... 目录一、环境准备二、基础调用:使用 Runtime.exec()2.1 实现步骤2.2 代码解析三、