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

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

相关文章

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

python运用requests模拟浏览器发送请求过程

《python运用requests模拟浏览器发送请求过程》模拟浏览器请求可选用requests处理静态内容,selenium应对动态页面,playwright支持高级自动化,设置代理和超时参数,根据需... 目录使用requests库模拟浏览器请求使用selenium自动化浏览器操作使用playwright

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

解密SQL查询语句执行的过程

《解密SQL查询语句执行的过程》文章讲解了SQL语句的执行流程,涵盖解析、优化、执行三个核心阶段,并介绍执行计划查看方法EXPLAIN,同时提出性能优化技巧如合理使用索引、避免SELECT*、JOIN... 目录1. SQL语句的基本结构2. SQL语句的执行过程3. SQL语句的执行计划4. 常见的性能优

linux下shell脚本启动jar包实现过程

《linux下shell脚本启动jar包实现过程》确保APP_NAME和LOG_FILE位于目录内,首次启动前需手动创建log文件夹,否则报错,此为个人经验,供参考,欢迎支持脚本之家... 目录linux下shell脚本启动jar包样例1样例2总结linux下shell脚本启动jar包样例1#!/bin

java内存泄漏排查过程及解决

《java内存泄漏排查过程及解决》公司某服务内存持续增长,疑似内存泄漏,未触发OOM,排查方法包括检查JVM配置、分析GC执行状态、导出堆内存快照并用IDEAProfiler工具定位大对象及代码... 目录内存泄漏内存问题排查1.查看JVM内存配置2.分析gc是否正常执行3.导出 dump 各种工具分析4.

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数