条款36:绝不重新定义继承而来的non-virtual函数

2024-01-08 16:12

本文主要是介绍条款36:绝不重新定义继承而来的non-virtual函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.前言

假设class D是由class B以public形式派生而来,class B定义有一个public成员函数mf,由于mf的参数和返回值都不重要,所以定义两者都为void。即:

class B{public:void mf();...
};class D:public B{...};

2.实例分析

在这里,虽然我们对B和D及mf都一无所知,但面对一个类型为D的对象x:、

D x;//x是一个类型为D的对象

如果以下行为:

B* pB=&x;//获得一个指针指向x
pB->mf();//经由该指针调用mf
不同以下行为:
D* pD=&x;//获得一个指针指向x
pD->mf();//经由该指针调用mf

为什么两种用法会不同呢,毕竟都是通过对象x调用成员函数mf。由于两者所调用的函数都相同,凭借的对象也相同。

但实际上是这样的,更明确地说:如果mf是个non_virtual函数而D定义有自己地mf版本,那就不是如此:

class D:public B{public:void mf();//遮掩了B::mf()...
};
pB->mf();//调用B::mf
pD->mf();//调用D::mf

造成产生这种现象地原因是non-virtual函数如B::mf和D::mf都是静态绑定。这表明pB被声明为一个pointer-to-B,通过pB调用地non-virtual函数永远是B所定义地版本,即使pB指向一个类型为“B派生之class”地对象

另一方面,virtual函数却是动态绑定,所以它们不受该问题地困扰。如果mf是个virtual函数,不论是通过pB或者pD调用mf,都会导致调用D::mf,因为pB和pD真正指向地是一个类型为D地对象

如果你正在编写class D并重新定义继承自class B地non-virtual函数mf,D对象很可能产生相互矛盾地不一致行为。更明确地说当mf被调用时,任何一个D对象都可能表现出B或者D地行为:决定因素不在对象自身,而在于指向该对象地指针当初声明地类型。reference也会展现和指针一样难以理解地行径。

现在如果重新定义mf,设计便会出现矛盾,如果D真有必要实现出与B不同地mf,并且每一个B对象真的必须使用B所提供地mf实现码,那么每个D都是一个B就不为真。既然如此D就不该以public形式继承B。另一方面,如果D真地必须以public方式继承B,并且如果真有需要实现出与B不同地mf,那么mf就无法为B反映出“不变性凌驾于特异性”的性质。既然这样mf应该声明为virtual函数。最后如果每个D真的是一个B,并且如果mf真的为B反映出“不变性凌驾于特异性”的性质上,那么D便不需要重新定义mf,而且它也不应该尝试这样做。

不论哪一个观点,结论都一样:任何情况下都不该重新定义一个继承而来的non-virtua函数。

这篇关于条款36:绝不重新定义继承而来的non-virtual函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

JavaSE——封装、继承和多态

1. 封装 1.1 概念      面向对象程序三大特性:封装、继承、多态 。而类和对象阶段,主要研究的就是封装特性。何为封装呢?简单来说就是套壳屏蔽细节 。     比如:对于电脑这样一个复杂的设备,提供给用户的就只是:开关机、通过键盘输入,显示器, USB 插孔等,让用户来和计算机进行交互,完成日常事务。但实际上:电脑真正工作的却是CPU 、显卡、内存等一些硬件元件。

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

OpenCV结构分析与形状描述符(11)椭圆拟合函数fitEllipse()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C++11 算法描述 围绕一组2D点拟合一个椭圆。 该函数计算出一个椭圆,该椭圆在最小二乘意义上最好地拟合一组2D点。它返回一个内切椭圆的旋转矩形。使用了由[90]描述的第一个算法。开发者应该注意,由于数据点靠近包含的 Mat 元素的边界,返回的椭圆/旋转矩形数据

Unity3D 运动之Move函数和translate

CharacterController.Move 移动 function Move (motion : Vector3) : CollisionFlags Description描述 A more complex move function taking absolute movement deltas. 一个更加复杂的运动函数,每次都绝对运动。 Attempts to

✨机器学习笔记(二)—— 线性回归、代价函数、梯度下降

1️⃣线性回归(linear regression) f w , b ( x ) = w x + b f_{w,b}(x) = wx + b fw,b​(x)=wx+b 🎈A linear regression model predicting house prices: 如图是机器学习通过监督学习运用线性回归模型来预测房价的例子,当房屋大小为1250 f e e t 2 feet^

JavaSE(十三)——函数式编程(Lambda表达式、方法引用、Stream流)

函数式编程 函数式编程 是 Java 8 引入的一个重要特性,它允许开发者以函数作为一等公民(first-class citizens)的方式编程,即函数可以作为参数传递给其他函数,也可以作为返回值。 这极大地提高了代码的可读性、可维护性和复用性。函数式编程的核心概念包括高阶函数、Lambda 表达式、函数式接口、流(Streams)和 Optional 类等。 函数式编程的核心是Lambda