C++:多态的内容和底层原理

这篇具有很好参考价值的文章主要介绍了C++:多态的内容和底层原理。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

本篇总结C++中多态的基本内容和原理实现和一些边角内容

多态的概念

首先要清楚多态是什么,是用来做什么的?

多态从字面意思来讲,就是多种形态,完成一个事情,不同的人去完成会有不同的结果和状态,这样的情况就叫做多态

多态的定义

多态是不同继承关系的类对象,在调用一个函数的时候会产生不同的行为,比如同样是买票这个操作,普通人就是全票,学生就是半票,本篇的例子也会从这个例子出发,进行多态中具体的距离和深层次的理解

构成多态的条件:

  1. 必须通过基类的指针或者引用调用虚函数
  2. 被调用的函数必须是虚函数,并且派生类要对基类的虚函数进行重写
#include <iostream>
using namespace std;

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

class Student :public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "学生票半价" << endl;
	}
};

void func(Person& p)
{
	p.BuyTicket();
}

int main()
{
	Person p;
	Student s;
	func(p);
	func(s);
}

上面是对多态的最初始定义,也是很基础的定义,从中可以看出多态的基本用法和实现的功能

虚函数

虚函数的定义:

虚函数通俗来说,就是被virtual修饰的类成员函数就是虚函数

虚函数的重写:

虚函数的重写就是,当派生类中有一个和基类完全相同的虚函数,那么就称之为子类的虚函数重写了基类的虚函数,虽然子类可以不加virtual,但是并不标准,最好加上

虚函数的例外:

  1. 协变
    协变就是,派生类重写基类虚函数的时候,与基类虚函数返回值类型不同,比如基类的虚函数返回的是基类成员的指针和引用,派生类返回的是指针和引用的时候,也算是虚函数重写,这种情况就叫做协变

  2. 析构函数重写
    如果基类的析构函数是虚函数,那么派生类的析构函数默认会和基类的析构函数构成重写,虽然名字和函数名不同,但是依旧是,这是因为编译器进行编译后,把析构函数的名称统一处理为destructor,这样也算是重写

class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "普通票全价" << endl;
	}
	virtual Person* f()
	{
		return new Person;
	}
	virtual ~Person()
	{
		cout << "~Person()" << endl;
	}
};

class Student :public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "学生票半价" << endl;
	}
	virtual Student* f()
	{
		return new Student;
	}
	virtual ~Student()
	{
		cout << "~Student()" << endl;
	}
};

void func(Person& p)
{
	p.BuyTicket();
}

override和final关键字

C++11中,引入了两个关键字,这两个关键字就是用来辅助进行虚函数多态的多种复杂情形,避免出现疏忽而导致错误的情况出现:

final:修饰虚函数,表示这个虚函数不能被重写了

class Student :public Person
{
public:
	virtual void BuyTicket() final
	{
		cout << "学生票半价" << endl;
	}
	virtual Student* f()
	{
		return new Student;
	}
	virtual ~Student()
	{
		cout << "~Student()" << endl;
	}
};

class Child :public Student
{
public:
	virtual void BuyTicket()
	{
		cout << "小孩免票" << endl;
	}
};

C++:多态的内容和底层原理,C++,知识总结,c++,开发语言,算法
override:检查派生类虚函数是否重写了基类某个虚函数,如果没有就报错

class Person
{
public:
	/*virtual*/ void BuyTicket()
	{
		cout << "普通票全价" << endl;
	}
	virtual Person* f()
	{
		return new Person;
	}
	virtual ~Person()
	{
		cout << "~Person()" << endl;
	}
};

class Student :public Person
{
public:
	virtual void BuyTicket() override
	{
		cout << "学生票半价" << endl;
	}
	virtual Student* f()
	{
		return new Student;
	}
	virtual ~Student()
	{
		cout << "~Student()" << endl;
	}
};

class Child :public Student
{
public:
	virtual void BuyTicket() override
	{
		cout << "小孩免票" << endl;
	}
};

void func(Person& p)
{
	p.BuyTicket();
}

C++:多态的内容和底层原理,C++,知识总结,c++,开发语言,算法
多态在使用的过程中是十分复杂的,因此使用时需要注意逻辑能否清楚的表示,可能只是稍微变了一点点内容,就使得整个意思全然变换,下面对比一下继承多态中的一些概念:

重载、覆盖、隐藏

  1. 重载指的是,函数名在一个作用域,并且函数名相同,参数不同的情况,那么这两个函数就构成了函数重载,编译器在进行处理的时候会根据参数形成不同的函数表,由此来对应不同的情况
  2. 重写指的是,两个函数在基类和派生类的作用域下,前提是函数名、参数、返回值都一样的情况下,如果是虚函数,那么就构成了重写,其中子类可以不写virtual,可以理解为虚函数的属性被从基类中继承了下来,但是并不推荐这样写,其中要注意特殊情况,比如协变和析构函数的情况
  3. 隐藏指的是,两个函数在基类和派生类的作用域下,当函数名相同的时候,如果不符合重写的定义那么就是重定义了,比如在继承中见到的很多种情况

抽象类

抽象类的定义

在虚函数后面写上等于0,就说明这个函数是纯虚函数,有纯虚函数的类就叫做抽象类,抽象类的特点是不可以实例化出一个具体的对象,而派生类被继承后也不能实例化对象,只有在重写了虚函数的前提下,才能实例化对象

纯虚函数体现了派生类要重写的这个规则,同时也体现出了接口继承的概念

接口继承和实现继承

  1. 接口继承(Interface Inheritance)是指从一个纯虚基类(pure virtual base class)继承而来,目的是为了实现一个类的接口,使得派生类必须实现该接口中定义的所有纯虚函数。接口继承的主要目的是实现类的接口复用,它并不关心实现细节。在接口继承中,派生类只需要实现基类中定义的纯虚函数,不需要关心基类中其他的数据和函数
class Shape 
{
public:
    virtual void draw() = 0; // 纯虚函数
};

class Circle : public Shape 
{
public:
    void draw() override 
    {
        // 实现圆形的绘制
    }
};

class Square : public Shape 
{
public:
    void draw() override 
    {
        // 实现正方形的绘制
    }
};
  1. 实现继承(Implementation Inheritance)是指从一个普通的基类(非纯虚基类)继承而来,目的是为了实现基类中已有的函数或数据。实现继承的主要目的是实现代码复用,它关心基类中的实现细节。在实现继承中,派生类会继承基类中所有的成员函数和数据成员,并且可以重写这些函数以改变它们的行为
class Person 
{
public:
    void sayHello() 
    {
        std::cout << "Hello, I am a person." << std::endl;
    }
};

class Student : public Person 
{
public:
    void sayHello() override 
    {
        std::cout << "Hello, I am a student." << std::endl;
    }
};

int main() 
{
    Student s;
    s.sayHello(); // 输出: "Hello, I am a student."
    return 0;
}

接口继承是指派生类只继承了基类的接口(也就是纯虚函数),而没有继承基类的实现。这种方式使得派生类必须实现基类中的所有纯虚函数,从而使得派生类和基类的实现是分离的,实现了接口和实现的分离。这种继承方式常常用于实现抽象类和接口,强制要求派生类实现接口中的所有函数

实现继承是指派生类继承了基类的接口和实现,包括数据成员和函数实现。这种方式使得派生类可以复用基类的代码,从而减少了代码的重复编写,同时也保证了派生类和基类的一致性。但是,这也意味着派生类和基类的实现是紧密耦合的,基类的修改可能会影响到派生类的行为

多态的原理解析

虚函数表

对于一个使用了多态的类,创建一个对象看其内部的内容:

C++:多态的内容和底层原理,C++,知识总结,c++,开发语言,算法
会发现这当中和预想的结果并不一样,原因就在于这当中多了一个数组,这个指针数组实际上是叫做虚函数表指针数组,严格意义来说,这个数组中存放的是虚函数的函数地址,虚函数表也叫做虚表,那么问题来了:为什么要这么设计呢?

下面来做实验,对前面的类进行改造:

Person类中加一个虚函数和一个普通函数,而在Student类中只重写一个虚函数:

class Person
{
public:
	// Person类中有两个虚函数和一个普通函数
	virtual void BuyTicket()
	{
		cout << "普通票全价" << endl;
	}
	virtual void func1()
	{
		cout << "void func1()" << endl;
	}
	void func2()
	{
		cout << "void func2()" << endl;
	}
private:
	int _person; // 定义一个变量
};

class Student :public Person
{
	// 继承类中只重写一个虚函数,剩下的不进行重写
public:
	virtual void BuyTicket()
	{
		cout << "学生票半价" << endl;
	}
private:
	int _student; // 定义一个变量
};

void func(Person& p)
{
	p.BuyTicket();
}

int main()
{
	Person p;
	Student s;
	func(p);
	func(s);
}

实验结果如下:

C++:多态的内容和底层原理,C++,知识总结,c++,开发语言,算法
对实验结果进行分析得出下面的结论:

  1. 派生类的Student对象中也有一个虚表指针,其中是由两个部分组成的,一个是父类成员和自己的成员,虚表指针中也是存在的一部分是自己的成员
  2. 在基类和派生类中的虚表地址是不一样的,但是在虚表的具体内部中会发现,有一个函数指针地址是一样的,还有一个不一样,那么说明在Student类中重写的函数发生了改变,因此虚函数的重写才叫做覆盖,覆盖指的就是虚表中对于虚函数的覆盖
  3. 对于虚表内的内容,只有被继承下来的虚函数才会放到虚表中,其余函数不会放入虚表中
  4. 虚函数表本质上就是一个存放虚函数指针的指针数组,这与一开始的结论是一样的

虚函数表的生成过程:

  1. 基类中的虚表拷贝到派生类的虚表中
  2. 如果派生类中重写了虚表的某个函数,那么就进行覆盖的过程
  3. 派生类自己新有的虚函数按照在类内的次序放到派生类虚表的最后

虚函数和虚表的存储位置:

虚函数存放在虚表,虚表存放在对象中,这样的回答是错误的!

虚表中存的是虚函数的指针,并不是虚函数,虚函数和普通的函数是一样的,都是存放在代码段中,只是将它的指针放到了虚表中,而在对象中存放的是虚表的指针,也不是虚表,虚表在vs下也是存放在代码段的位置中

多态的调用原理:

在知道了虚表的存在和原理后,其实可以理解前面的一些内容了

当指向的对象是Person类的时候,此时会在Person类的虚表中找到对应的函数并进行调用,当对象是Student类的时候,原理相同,借助这个原理就实现了多态,用不同的对象去运行会产生不同的结果,而多态的函数调用也不是直接确认的,而是在运行的过程中,在对象的内部自动取识别,去获取的

动态绑定和静态绑定

静态绑定也叫做前期绑定或者是早绑定:在程序编译期间就确定了程序的行为,也叫做静态多态,比如说函数重载就是比较典型的例子

动态绑定也叫做后期绑定或者是晚绑定:在程序运行期间,根据具体拿到的类型来确定程序的具体行为和调用的具体函数,比如说动态多态就是这样的例子文章来源地址https://www.toymoban.com/news/detail-718979.html

到了这里,关于C++:多态的内容和底层原理的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++类开发第七篇(详细说说多态和编译原理)

    多态性(polymorphism)提供接口与具体实现之间的另一层隔离,从而将”what”和”how”分离开来。多态性改善了代码的可读性和组织性,同时也使创建的程序具有可扩展性,项目不仅在最初创建时期可以扩展,而且当项目在需要有新的功能时也能扩展。 c++支持编译时多态(静态多

    2024年03月09日
    浏览(53)
  • 【C++】多态原理剖析,Visual Studio开发人员工具使用查看类结构cl /d1 reportSingleClassLayout

    author:Carlton tag:C++ topic:【C++】多态原理剖析,Visual Studio开发人员工具使用查看类结构cl /d1 reportSingleClassLayout website:黑马程序员C++ tool:Visual Studio 2019 date:2023年7月24日   目录 父类使用虚函数前后类内部结构变化 子类重写父类虚函数的作用及其机理         首先父类成员

    2024年02月15日
    浏览(75)
  • C++中菱形继承中的多态在底层是如何实现的。

    如果还不了解菱形继承和多态的底层可以看这两篇文章: C++中多态的底层实现_Qianxueban的博客-CSDN博客 C++的继承以及virtual的底层实现_Qianxueban的博客-CSDN博客

    2024年02月09日
    浏览(39)
  • [C++] 多态(下) -- 多态原理 -- 动静态绑定

    上一篇文章我们了解了虚函数表,虚函数表指针,本篇文章我们来了解多态的底层原理,更好的理解多态的机制。 [C++] 多态(上) – 抽象类、虚函数、虚函数表 下面这段代码中,Func函数传Person调用的Person::BuyTicket,传Student调用的是Student::BuyTicket,这就是多态调用,但是这里我

    2024年02月04日
    浏览(49)
  • 波奇学C++:多态知识点

    结果是 student 0 原因在于重写时只重写函数的实现,就是说相当于Person的fun的声明和Student的函数实现的拼在一起所以缺省值是0。 如果是子类指针或者引用就不是多态调用了只是单存子类对父类的重定义,隐藏函数。 上一篇文章提到的,多态的本质就是基类和派生类的虚表中

    2024年02月09日
    浏览(41)
  • C++修炼之路之多态---多态的原理(虚函数表)

    目录 一:多态的原理  1.虚函数表  2.原理分析 3.对于虚表存在哪里的探讨 4.对于是不是所有的虚函数都要存进虚函数表的探讨 二:多继承中的虚函数表 三:常见的问答题  接下来的日子会顺顺利利,万事胜意,生活明朗-----------林辞忧  接上篇的多态的介绍后,接下来介绍

    2024年04月26日
    浏览(44)
  • 【C++】多态及原理

    1.多态的概念 多态,顾名思义就是多种状态, 具体点就是去完成某种行为,但是通过不同的对象去完成某种行为都会产生不同的状态,这就是多态 比如买票这个行为。当 普通人 买票时,是全价买票; 学生 买票时,是半价买票; 军人 买票时是优 先买票。 这就是不同的对象

    2024年02月16日
    浏览(35)
  • C++中多态的原理

    上篇讲解了多态的原理,这篇文章来详细讲解一下多态的原理。 这里有一道常考笔试题:sizeof(Base)是多少? 为什么不是8? 可以调试带大家看一下。 仔细看,对象的头部多了一个指针。 这个指针叫做虚函数表指针。 上面不重要,重要的是下面的东西,多态的原理。 这个指

    2024年02月04日
    浏览(43)
  • C++中的多态你真的了解吗?多态原理全面具体讲解

    目录 1. 多态的概念 2. 多态的定义及实现 2.1 多态的构成条件 2.2 虚函数 2.3 虚函数的重写 2.4 C++11 override 和 final 2.5 重载、覆盖(重写)、隐藏(重定义)的对比 3. 抽象类 3.1 概念 4. 多态的原理 4.1 虚函数表 4.2多态的原理 4.3 动态绑定与静态绑定 5. 单继承和多继承关系中的虚函数表

    2024年02月04日
    浏览(34)
  • [C++]:万字超详细讲解多态以及多态的实现原理(面试的必考的c++考点)

    文章目录 前言 一、多态的定义及实现 1.多态的构成条件 2.c++11的override和final 3.重载,重写,重定义的比较 4.抽象类 5.多态的原理 6.多继承中的虚函数表 7.动态绑定和静态绑定 总结 多态的概念: 多态的概念:通俗来说,就是多种形态, 具体点就是去完成某个行为,当不同的

    2023年04月22日
    浏览(62)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包