首页 > 编程语言 >【C++从小白到大牛】多态那些事儿(上)

【C++从小白到大牛】多态那些事儿(上)

时间:2024-08-07 10:23:55浏览次数:14  
标签:调用 函数 子类 多态 C++ 父类 重写 白到

目录

一、多态的概念

1.1概念:

二、 多态的定义及实现

 2.1多态的构成条件

在继承中要多态还要两个条件

 2.2虚函数

2.3虚函数的重写

2.4虚函数重写的两个例外:

1. 协变(基类与派生类虚函数返回值类型不同)(了解 不重要)

2. 析构函数的重写(基类与派生类析构函数的名字不同)

2.5、关于父类子类virtual加与不加

2.6 C++11 override 和 final

2.7重载、覆盖(重写)、隐藏(重定义)的对比

四、多态的原理

4.1虚函数表

面试题解析:


一、多态的概念

1.1概念:

通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态。

二、 多态的定义及实现

 2.1多态的构成条件

多态是在不同继承关系的类对象,去调用同一函数,产生了不同的行为。比如Student继承了Person。Person对象买票全价,Student对象买票半价。

在继承中要多态还要两个条件

  1. 父类指针或引用去调用虚函数,这样才能保证传父类对象调用的就是父类的虚函数,传子类对象,调用的是子类的虚函数(调用子类传子类利用切片的原理)
  2. 虚函数完成重写:父子类中的两个虚函数,三同(函数名、参数、返回值)这样父子类的两个虚函数才能构成重写

可以将重写理解为隐藏的子集,因为隐藏仅要求函数名相同

 2.2虚函数

虚函数:即被virtual修饰的类成员函数称为虚函数。

class Person {
public:
 virtual void BuyTicket() { cout << "买票-全价" << endl;}
};

2.3虚函数的重写

虚函数的重写(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称子类的虚函数重写了基类的虚函数。

class Person {
public:
 virtual void BuyTicket() { cout << "买票-全价" << endl; }
};
class Student : public Person {
public:
 virtual void BuyTicket() { cout << "买票-半价" << endl; }
/*注意:在重写基类虚函数时,派生类的虚函数在不加virtual关键字时,虽然也可以构成重写(因
为继承后基类的虚函数被继承下来了在派生类依旧保持虚函数属性),但是该种写法不是很规范,不建议这样使用*/
/*void BuyTicket() { cout << "买票-半价" << endl; }*/
};
void Func(Person& p)
{ p.BuyTicket(); }

2.4虚函数重写的两个例外:

1. 协变(基类与派生类虚函数返回值类型不同)(了解 不重要)

协变,虚函数返回值可以不同,返回值要求必须是父子类关系的指针或者引用。

class A{};
class B : public A {};
class Person {
public:
 virtual A* f() {return new A;}
};
class Student : public Person {
public:
 virtual B* f() {return new B;}
};

2. 析构函数的重写(基类与派生类析构函数的名字不同)

普通调用:看指针或者引用或者对象的类型

多态调用:看指针或者引用指向的对象

我们希望上面的特殊情况是多态调用,如果是普通调用,会造成内存泄漏(student对象没有析构,如下图)。

所以我们如何才能变成多态调用呢?

在子类和基类的析构函数都加上virtual构成重写,变为多态调用。那这里违反重写的规则,函数名都不相同,怎么能构成重写呢?

答:

虽然函数名不相同,看起来违背了重写的规则,其实不然,这里可以理解为编译器对析构函数的名称做了特殊处理,编译后析构函数的名称统一处理成destructor!

TIP:特殊情况

如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual关键字,都与基类的析构函数构成重写。

2.5、关于父类子类virtual加与不加

虚函数重写时,父类虚函数加了virtual,子类不加,也构成重写;但是如果父类不加virtual,就不构成重写。

建议:两个虚函数都加上virtual

2.6 C++11 override 和 final

final 关键字的两个作用:

其一是final修饰的类是最终类,不能被继承

注意实现一个类,这个类不能被继承还有一种方法:让父类构造函数私有化,派生类实例化不出对象。

其二是修饰虚函数,表示该虚函数不能再被重写

override: 检查派生类虚函数是否重写了基类某个虚函数,如果没有重写编译就会报错。

2.7重载、覆盖(重写)、隐藏(重定义)的对比

重载:

  1. 两个函数在同一作用域
  2. 函数名相同,参数不同

重写(覆盖)

  1. 两个函数分别在基类和派生类的作用域
  2. 函数名/参数/返回值都必须相同(协变例外)
  3. 两个函数都必须是虚函数

重定义(隐藏)

  1. 两个函数分别在基类和派生类的作用域
  2. 函数名相同
  3. 两个基类和派生类的同名函数不构成重写就是重定义

四、多态的原理

4.1虚函数表

先做一道笔试题:

// 这里常考一道笔试题:sizeof(Base)是多少?
class Base
{
public:
 virtual void Func1()
 {
 cout << "Func1()" << endl;
 }
private:
 int _b = 1;
};

通过观察测试我们发现b对象是8bytes。

除了_b成员,还多一个__vfptr指针,也就是虚函数表指针,这个表本质上是一个函数指针的数组

vfptr是存放函数指针的数组,就是将虚函数的指针存进去。

虚函数的重写也叫做覆盖,重写是语法层的概念,覆盖是原理层的概念。

形象的记忆:

比如上图,子类将父类的拷贝过来,然后虚表重写的部分将原先父类虚表的部分进行一个覆盖。

具体如何实现多态调用和普通调用?

  • 多态调用:

运行时去虚函数表中找函数的地址,进行调用,所以指向父类调用的是父类虚函数,指向子类调用的是子类虚函数。

  • 普通调用:

编译时,通过调用者类型确定函数地址。

面试题解析:

解析:

首先我们看到B继承A,那么B里面的func函数和A里面的func函数构不构成重写呢?

函数名相同,返回值相同,参数类型相同(注意看参数是否相同,就是看类型,与变量名、缺省值无关!)并且父类是虚函数,所以构成重写!

接着我们看到p->test(),直接调用到了父类test()里面的func()函数,那么这里this指针是A* or B*呢,因为此时的test()是在父类,因此是A*,(如果是B*子类,那就不满足多态的条件必须是父类,所以不满足多态)所以这里的func函数构成了多态,因此是多态调用,所以是指针/引用指向的类型,因此调用B里面的func函数,所以答案是D嘛?

但真正的答案是B。

原因是多态调用,重写是实现重写,会将父类的函数声明与子类进行组合,因此val的值就是父类的1,因此答案是B!

下面因为是子类的调用,不构成多态,因此答案全都是D

标签:调用,函数,子类,多态,C++,父类,重写,白到
From: https://blog.csdn.net/hanwangyyds/article/details/140980181

相关文章

  • C++入门基础1
    目录1.c++发展历史2.C++在⼯作领域中的应⽤3.C++学习建议和书籍推荐3.1学习难度3.2书籍的推荐4.c++第一个程序5.命名空间5.1namesapce的价值5.2namespace的定义5.2.1 正常的命名空间定义5.3命名空间的使用5.3.1指定命名空间访问5.3.2using将命名空间中某......
  • C++笔记,类和对象(上)
    对于类的初步认识目录对于类的初步认识(1)类的定义(2)类的访问限定符及封装(3)类的作用域(4)类的实例化(5)类的对象大小的计算(6)类成员函数的this指针(1)类的定义classclassName{//类体,由成员函数和成员变量组成};//一定要注意后面的分号类体中内容称为类的成员:类......
  • Qt/C++最新地图组件发布/历时半年重构/同时支持各种地图内核/包括百度高德腾讯天地图
    一、前言说明最近花了半年时间,专门重构了整个地图组件,之前写的比较粗糙,有点为了完成功能而做的,没有考虑太多拓展性和易用性。这套地图自检这几年大量的实际项目和用户使用下来,反馈了不少很好的建议和意见,经过这几年的整理,刚好趁着近期经济下行严重,抽出时间把整个地图组件重构一下......
  • C++ 学习预备知识
    1C++简介 1.1起源    C++与C语言一样,也是在贝尔实验室诞生的,名称C++来自C语言中的递增运算符++,该运算符将变量加1。这也表明,C++是C语言的扩充版本。    C++融合了3种不同的编程方式:C语言代表的过程性语言、C++在C语言基础上添加的类代表的面向对象语言、C+......
  • C++解析ini文件
    目录一.什么是ini文件二.ini文件的格式一般是什么样的1.节2.参数3.注释三.C++实现ini文件的解析四.其他这篇文章简单讨论一下ini文件。一.什么是ini文件ini文件其实就是一种配置文件,常见于Windows的系统配置文件,当然也可以是其他用途,你的用法你说了算。二.ini文件......
  • Java SE知识点六:面向对象之:多态
    1.多态的概念多态也是Java面向对象的三大要点之一,多态的概念通俗来讲就是一个事物的多种形态,让不同的对象去完成一个方法时会出现不同的状态。比如,猫吃饭和狗吃饭就是不同的状态,但用到的方法都是一样。2.实现多态2.1实现条件要实现多态需要遵循以下条件:必须是在......
  • Java 类多态的向上转型
     假定Base b = new Derived(); 调用执行b.methodOne()后,输出结果是什么?1publicclassBase2{3publicvoidmethodOne()4{5System.out.print("A");6methodTwo();7}89publicvoidmethodTwo()10{11System.ou......
  • 【C++/STL】map和set的封装(红黑树)
     ......
  • Java 继承和多态(进阶介绍 十六)
    目录Java继承IS-A关系实例实例instanceof关键字HAS-A关系例子Java多态简单的例子虚方法多态的实现方式方式一:重写方式二:接口Java继承继承是所有OOP语言和Java语言不可缺少的组成部分。继承是Java面向对象编程技术的一块基石,是面向对象的三大特征......
  • C++学习笔记----Strings与String View(4)-- 字符串操作
        今天讲点简单易懂的,字符串操作,当然了,不是全部,列出几个典型的字符串操作,完整地可以参考相关资料,网上一搜一把哦。substr(pos,len):返回特定位置pos,特定长度的子字符串。find(str):返回字符串的位置,如未找到则返回string::npos。replace(pos,len,str):用新的字符串str......