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

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

相关文章

SpringBoot 整合 Grizzly的过程

《SpringBoot整合Grizzly的过程》Grizzly是一个高性能的、异步的、非阻塞的HTTP服务器框架,它可以与SpringBoot一起提供比传统的Tomcat或Jet... 目录为什么选择 Grizzly?Spring Boot + Grizzly 整合的优势添加依赖自定义 Grizzly 作为

mysql-8.0.30压缩包版安装和配置MySQL环境过程

《mysql-8.0.30压缩包版安装和配置MySQL环境过程》该文章介绍了如何在Windows系统中下载、安装和配置MySQL数据库,包括下载地址、解压文件、创建和配置my.ini文件、设置环境变量... 目录压缩包安装配置下载配置环境变量下载和初始化总结压缩包安装配置下载下载地址:https://d

Idea调用WebService的关键步骤和注意事项

《Idea调用WebService的关键步骤和注意事项》:本文主要介绍如何在Idea中调用WebService,包括理解WebService的基本概念、获取WSDL文件、阅读和理解WSDL文件、选... 目录前言一、理解WebService的基本概念二、获取WSDL文件三、阅读和理解WSDL文件四、选择对接

springboot整合gateway的详细过程

《springboot整合gateway的详细过程》本文介绍了如何配置和使用SpringCloudGateway构建一个API网关,通过实例代码介绍了springboot整合gateway的过程,需要... 目录1. 添加依赖2. 配置网关路由3. 启用Eureka客户端(可选)4. 创建主应用类5. 自定

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

最新版IDEA配置 Tomcat的详细过程

《最新版IDEA配置Tomcat的详细过程》本文介绍如何在IDEA中配置Tomcat服务器,并创建Web项目,首先检查Tomcat是否安装完成,然后在IDEA中创建Web项目并添加Web结构,接着,... 目录配置tomcat第一步,先给项目添加Web结构查看端口号配置tomcat    先检查自己的to

SpringBoot集成SOL链的详细过程

《SpringBoot集成SOL链的详细过程》Solanaj是一个用于与Solana区块链交互的Java库,它为Java开发者提供了一套功能丰富的API,使得在Java环境中可以轻松构建与Solana... 目录一、什么是solanaj?二、Pom依赖三、主要类3.1 RpcClient3.2 Public

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

SpringBoot整合kaptcha验证码过程(复制粘贴即可用)

《SpringBoot整合kaptcha验证码过程(复制粘贴即可用)》本文介绍了如何在SpringBoot项目中整合Kaptcha验证码实现,通过配置和编写相应的Controller、工具类以及前端页... 目录SpringBoot整合kaptcha验证码程序目录参考有两种方式在springboot中使用k