【C++研发面试笔记】2. 多态性

2024-04-19 15:48

本文主要是介绍【C++研发面试笔记】2. 多态性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【C++研发面试笔记】2. 多态性

2.1 多态性来源

多态性指相同对象收到不同消息或不同对象收到相同消息时产生不同的实现动作。
最常见的用法就是声明基类的指针,利用该指针指向任意一个子类对象,调用相应的虚函数,可以根据指向的子类的不同而实现不同的方法。如果没有使用虚函数的话,即没有利用C++多态性,则利用基类指针调用相应的函数的时候,将总被限制在基类函数本身,而无法调用到子类中被重写过的函数。
C++支持两种多态性:编译时多态性,运行时多态性。
a. 编译时多态性:通过函数重载、重写、模板来实现。
b. 运行时多态性:通过虚函数继承实现。对于虚函数来说,父类和子类都有各自的版本。由多态方式调用的时候动态绑定。

重载、重写与屏蔽的区别
重载:在相同作用域内,函数名称相同,参数或常量性不同的相关函数称为重载。
重写:派生类重写基类中同名同参数同返回值的函数(通常是虚函数,这是推荐的做法)。
屏蔽:一个内部作用域(派生类,嵌套类或名字空间)内提供一个同名但不同参数或不同常量性的函数,使得外围作用域的同名函数在内部作用域不可见,编译器在进行名字查找时将在内部作用域找到该名字从而停止去外围作用域查找,因而屏蔽外围作用域的同名函数。尽量不要屏蔽外围作用域(包括继承而来的)名字。屏蔽所带来的隐晦难以理解。


2.2 继承、接口与组合

2.2.1 派生类的3种继承方式

(1)公有继承方式:基类成员对其对象的可见性与一般类及其对象的可见性相同,公有成员可见,其他成员不可见。这里保护成员与私有成员相同。
(2)私有继承方式:基类成员对其对象的可见性与一般类及其对象的可见性相同,公有成员可见,其他成员不可见。基类成员对派生类的可见性对派生类来说,基类的公有成员和保护成员可见,基类的公有成员和保护成员都作为派生类的私有成员,并且不能被这个派生类的子类所访问;基类的私有成员不可见,派生类不可访问基类中的私有成员。尽量避免 private 继承,因为从基类继承而来的所有接口均为私有的,外部不可访问。
(3)保护继承方式:这种继承方式与私有继承方式的情况相同,两者的区别仅在于对派生类的成员而言,基类成员对其对象的可见性与一般类及其对象的可见性相同,公有成员可见,其他成员不可见。
(4)虚拟继承是多重继承中特有的概念。虚拟基类是为了解决多重继承而出现的,如类D继承自类B和类C,而类B和类C都继承自类A,此时类D中会继承两次A。为了节省内存空间,可以将B、C对A的继承定义为虚拟继承,而A成了虚拟基类,此时只需要继承一次。

2.2.2 继承的实质

(1)覆盖

  • 构造函数从最初始的基类开始构造的,所以对于某个子类具体实现,其也会默认构造全体父系类。派生类的内存大小=派生类本身数据变量+父类大小+虚函数表指针+指向父类指针。各个父类的同名变量不会形成覆盖,都是单独的变量。
  • 覆盖函数的就近调用,如果父类存在相关接口则优先调用父类接口(此时操作的是父类实例),如果父类也不存在相关接口则调用祖父辈接口。
  • 析构函数也是从子类开始向上析构的。假设有如下情况,带非虚析构函数的基类指针 pb 指向一个派生类对象d,而派生类在其析构函数中释放了一些资源,如果我们 delete pb;那么派生类对象的析构函数就不会被调用,从而导致资源泄漏发生。因此,应该声明基类的析构函数为虚函数。

(2)重写
在继承中有三类函数:纯虚函数、虚函数、非虚函数。

  • 非虚函数:当一个成员函数为非虚函数时,它在派生类中的行为就不应该不同。实际上,非虚成员函数表明了一种特殊性上的不变性,因为它表示的是不会改变的行为。所以,声明非虚函数的目的在于,使派生类强制继承函数的接口和实现。
  • 纯虚函数:声明纯虚函数的目的是让派生类来继承函数接口而不是实现,而实现交由派生类来完成,派生类也必须重写该接口(如果要实例的话)。
  • 虚函数:让派生类来继承函数接口和实现,而派生类可以重写该接口,但也可以调用基类的默认实现。
2.2.3 继承与组合间的区别:

面向对象编程讲究的是代码复用,继承和组合都是代码复用的有效方法。组合是将其他类的对象作为成员使用,继承是子类可以使用父类的成员方法。
继承在继承结构中,父类的内部细节对于子类是可见的。简单易用,使用语法关键字即可轻易实现。易于修改或扩展那些父类被子类复用的实现。编译阶段静态决定了层次结构,不能在运行期间进行改变。破坏了封装性,由于“白盒”复用,父类的内部细节对于子类而言通常是可见的。子类与父类之间紧密耦合,子类依赖于父类的实现,子类缺乏独立性。当父类的实现更改时,子类也不得不会随之更改。
组合是通过对现有的对象进行拼装(组合)产生新的、更复杂的功能。因为在对象之间,各自的内部细节是不可见的,所以我们也说这种方式的代码复用是“黑盒式代码复用”。通过获取指向其它的具有相同类型的对象引用,可以在运行期间动态地定义(对象的)组合。不破坏封装,整体类与局部类之间松耦合,彼此相对独立。
继承体现的是一种专门化的概念而组合则是一种组装的概念。除非用到向上转型,不然优先考虑组合。

2.2.4 继承与接口的区别:

首先来说接口也是一种继承方式,所谓接口实际上指的只有声明,对应的是只有纯虚函数的抽象类,在C++中并没有关于接口的关键字(这点同Java是不一样的)。


2.3 模板与多态

模板(泛型)是一种对类型进行参数化的工具,通常有两种形式:函数模板和类模板。函数模板针对仅参数类型不同的函数。类模板针对仅数据成员和成员函数类型不同的类。使用模板的目的就是能够让程序员编写与类型无关的代码。
注意:模板的声明或定义只能在全局,命名空间或类范围内进行。即不能在局部范围,函数内进行,比如不能在main函数中声明或定义一个模板。

2.3.1函数模板通式
// 模板函数定义
template < class 形参名 ... > 返回类型 函数名(参数列表)
{函数体}
// 比如定义交换函数
template <class T> void swap(T& a, T& b){};

其中template和class是关键字,class可以用typename关键字代替,在这里typename 和class没区别,<>括号中的参数叫模板形参,模板形参和函数形参很相像,模板形参不能为空。

2.3.2类模板通式
template < class 形参名... >  class 类名
{ ... };

类模板和函数模板都是以template开始后接模板形参列表组成,模板形参不能为空,一但声明了类模板就可以用类模板的形参名声明类中的成员变量和成员函数,即可以在类中使用内置类型的地方都可以使用模板形参名来声明。比如

template < class T> class A{
public: T a; T b;T hy(T c, T &d);
};

在类A中声明了两个类型为T的成员变量a和b,还声明了一个返回类型为T带两个参数类型为T的函数hy。
类模板对象的创建:比如一个模板类A,则使用类模板创建对象的方法为A<int> m;在类A后面跟上一个<>尖括号并在里面填上相应的类型,这样的话类A中凡是用到模板形参的地方都会被int 所代替。当类模板有两个模板形参时创建对象的方法为A<int, double> m;类型之间用逗号隔开。
在类模板外部定义成员函数的方法为:

template<模板形参列表> 返回类型 类名<模板形参名>::函数名(参数列表){函数体};
template < class T1,class T2> void A< T1,T2 >::h(){}
2.3.3 非类型形参

1、非类型模板形参:模板的非类型形参也就是内置类型形参,如template<class T, int a> class B{};其中int a就是非类型的模板形参。
2、 非类型形参在模板定义的内部是常量值,也就是说非类型形参在模板的内部是常量。
3、非类型模板的形参只能是整型、指针和引用,像double,String, String **这样的类型是不允许的。但是double &,double *对象的引用或指针是正确的。

2.3.4 操作符重载

运算符重载是对已有的运算符赋予多重含义,使同一个运算符作用于不同类型的数据导致不同类型的行为。
运算符重载的实质是函数重载。在实现过程中,首先把指定的运算表达式转化为对运算符函数的调用,运算对象转化为运算符函数的实参,然后根据实参的类型来确定需要调用达标函数。
不能重载的运算符只有五个,它们是:成员运算符“.”、指针运算符“*”、作用域运算符“::”、“sizeof”、条件运算符“?:”。
运算符重载形式有两种,重载为类的成员函数和重载为类的友元函数。
运算符重载为类的成员函数的一般语法形式为:

函数类型 operator 运算符(形参表) 
{ 函数体; 
} 

运算符重载为类的友元函数的一般语法形式为:

friend 函数类型 operator 运算符(形参表)
{ 函数体; 
}
2.3.5 拷贝构造函数

拷贝构造函数嘛,当然就是拷贝和构造了,是一种特殊的构造函数,它由编译器调用来完成一些基于同一类的其他对象的构建及初始化。代码结构如下:

Class X{
public:X();X(const X&);//拷贝构造函数
}

如果在类中没有显式地声明一个拷贝构造函数,那么,编译器将会自动生成一个默认的拷贝构造函数,该构造函数完成对象之间的浅拷贝自定义拷贝构造函数是一种良好的编程风格,它可以阻止编译器形成默认的拷贝构造函数,提高源码效率。
深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,只是定义了新的指针,就是浅拷贝。

2.3.6 C语言的泛型处理

在C语言中可以通过函数指针来实现部分泛型
C语言的泛型处理


2.4 动态绑定和静态绑定

1、静态对象:对象在声明时采用的类型。是在编译期确定的。
2、动态对象:指对象当前的类型,是在运行期决定的。对象的动态类型可以更改,但是静态类型无法更改。关于对象的静态类型和动态类型,看一个示例:

class B  
{  
}  
class C : public B  
{  
}  
class D : public B  
{  
}  
D* pD = new D();//pD的静态类型是它声明的类型D*,动态类型也是D*  
B* pB = pD;//pB的静态类型是它声明的类型B*,动态类型是pB所指向的对象pD的类型D*  
C* pC = new C();  
pB = pC;//pB的动态类型是可以更改的,现在它的动态类型是C*  

3、静态绑定:绑定的是对象的静态类型,某特性(比如函数)依赖于对象的静态类型,发生在编译期。
4、动态绑定:绑定的是对象的动态类型,某特性(比如函数)依赖于对象的动态类型,发生在运行期。

class B  
{  void DoSomething();  virtual void vfun();  
}  
class C : public B  
{  void DoSomething();//首先说明一下,这个子类重新定义了父类的no-virtual函数,这是一个不好的设计,会导致名称遮掩;这里只是为了说明动态绑定和静态绑定才这样使用。  virtual void vfun();  
}  
class D : public B  
{  void DoSomething();  virtual void vfun();  
}  
D* pD = new D();  
B* pB = pD;  

在这里,虽然pD和pB都指向同一个对象,但pD->DoSomething()和pB->DoSomething()并不是调用同一个函数。因为函数DoSomething是一个no-virtual函数,它是静态绑定的,也就是编译器会在编译期根据对象的静态类型来选择函数。pD的静态类型是D*,那么编译器在处理pD->DoSomething()的时候会将它指向D::DoSomething()。同理,pB的静态类型是B*,那pB->DoSomething()调用的就是B::DoSomething()。
而pD->vfun()和pB->vfun()调用的是同一个函数。因为vfun是一个虚函数,它动态绑定的,也就是说它绑定的是对象的动态类型,pB和pD虽然静态类型不同,但是他们同时指向一个对象,他们的动态类型是相同的,都是D*,所以,他们的调用的是同一个函数:D::vfun()。
上面都是针对对象指针的情况,对于引用(reference)的情况同样适用。
指针和引用的动态类型和静态类型可能会不一致,但是对象的动态类型和静态类型是一致的。D D; D.DoSomething()和D.vfun()永远调用的都是D::DoSomething()和D::vfun()。
总结一句话,只有虚函数才使用的是动态绑定,其他的全部是静态绑定。
特别需要注意的地方
当缺省参数和虚函数一起出现的时候情况有点复杂,因为虚函数是动态绑定的,而为了执行效率,缺省参数是静态绑定的。 所以绝不重新定义继承而来的缺省参数。


2.5 多继承的对象结构

已知:
class ClassC : public ClassA,public ClassB
下面这张图说明多继承下的对象结构:
这里写图片描述

这篇博文是个人的学习笔记,内容许多来源于网络(包括CSDN、博客园及百度百科等),博主主要做了微不足道的整理工作。由于在做笔记的时候没有注明来源,所以如果有作者看到上述文字中有自己的原创内容,请私信本人修改或注明来源,非常感谢>_<

这篇关于【C++研发面试笔记】2. 多态性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

跨国公司撤出在华研发中心的启示:中国IT产业的挑战与机遇

近日,IBM中国宣布撤出在华的两大研发中心,这一决定在IT行业引发了广泛的讨论和关注。跨国公司在华研发中心的撤出,不仅对众多IT从业者的职业发展带来了直接的冲击,也引发了人们对全球化背景下中国IT产业竞争力和未来发展方向的深思。面对这一突如其来的变化,我们应如何看待跨国公司的决策?中国IT人才又该如何应对?中国IT产业将何去何从?本文将围绕这些问题展开探讨。 跨国公司撤出的背景与

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象