当前位置: 首页 > news >正文

《C++多态详解:从虚函数到运行时动态绑定》

《C++多态详解:从虚函数到运行时动态绑定》


文章目录

  • 《C++多态详解:从虚函数到运行时动态绑定》
  • 一、多态的概念
  • 二、多态的定义及实现
    • 2.1 实现多态的两个必要条件
    • 2.2 虚函数
    • 2.3 虚函数的重写/覆盖
    • 2.4 多态场景的一个选择题
    • 2.5 虚函数重写的其他问题
      • 2.5.1 协变
      • 2.5.2 析构函数的重写
    • 2.6 override和final关键字
    • 2.7 重载/重写/隐藏的对比
  • 三、纯虚函数和抽象类
  • 四、多态的原理
    • 4.1 虚函数表指针
    • 4.2 多态的原理
      • 4.2.1 多态是如何实现
      • 4.2.2 动态绑定与静态绑定
      • 4.2.3 虚函数表
  • 五、整体源代码


一、多态的概念

在这里插入图片描述


二、多态的定义及实现

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


2.1 实现多态的两个必要条件

在这里插入图片描述

在这里插入图片描述


在这里插入图片描述


2.2 虚函数

类成员函数前面加virtual修饰,那么这个成员函数被称为虚函数。
注意:非成员函数不能加virtual修饰

在这里插入图片描述


2.3 虚函数的重写/覆盖

在这里插入图片描述在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


2.4 多态场景的一个选择题

本质重写的是基类的函数实现部分,函数声明部分编译认为子类和父类是一样的,
所以就默认使用父类

满足多态之后,用父类的声明部分+派生类重写的实现部分
在这里插入图片描述
在这里插入图片描述


在这里插入图片描述


2.5 虚函数重写的其他问题

2.5.1 协变

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


2.5.2 析构函数的重写

在这里插入图片描述


这里注意:析构函数加不加virtual都构成重写!
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


在这里插入图片描述


2.6 override和final关键字

在这里插入图片描述在这里插入图片描述

在这里插入图片描述
在这里插入图片描述


2.7 重载/重写/隐藏的对比

共同点:在这里插入图片描述

在这里插入图片描述


三、纯虚函数和抽象类

在这里插入图片描述在这里插入图片描述


四、多态的原理

4.1 虚函数表指针

在这里插入图片描述

在这里插入图片描述在这里插入图片描述


在这里插入图片描述在这里插入图片描述


4.2 多态的原理

在这里插入图片描述

4.2.1 多态是如何实现

从底层的⻆度Func函数中ptr->BuyTicket(),是如何作为ptr指向Person对象调⽤Person::BuyTicket,ptr指向Student对象调⽤Student::BuyTicket的呢?
在这里插入图片描述在这里插入图片描述

在这里插入图片描述


注意:ptr指向的始终存的都是Person对象
有可能本身就是Person对象,也有可能是切片出来的Person对象

在这里插入图片描述在这里插入图片描述
在这里插入图片描述


4.2.2 动态绑定与静态绑定

静态绑定效率会高一些!
在这里插入图片描述在这里插入图片描述
在这里插入图片描述


4.2.3 虚函数表

在这里插入图片描述在这里插入图片描述


在这里插入图片描述在这里插入图片描述
在这里插入图片描述


在这里插入图片描述在这里插入图片描述
g++里虚函数表存在了 静态区!


五、整体源代码

代码如下(示例):

// 多态!!!
//int main()
//{
//	int i;
//	double d;
//	cout << i;
//	cout << d;
//}
//
//swap(i1, i2);
//swap(d1, d2);//class Person {
//public:
//	virtual void BuyTicket() { cout << "买票-全价" << endl; }
//};
//class Student : public Person {
//public:
//	virtual void BuyTicket() { cout << "买票-打折" << endl; }
//};
//void Func(Person* ptr)
//{
//	// 这⾥可以看到虽然都是Person指针Ptr在调⽤BuyTicket
//	// 但是跟ptr没关系,⽽是由ptr指向的对象决定的。
//	ptr->BuyTicket();
//}
//int main()
//{
//	Person ps;
//	Student st;
//	Func(&ps);
//	Func(&st);
//	return 0;
//}//class Animal
//{
//public:
//	virtual void talk() const
//	{}
//};
//class Dog : public Animal
//{
//public:
//	virtual void talk() const
//	{
//		std::cout << "汪汪" << std::endl;
//	}
//};
//class Cat : public Animal
//{
//public:
//	virtual void talk() const
//	{
//		std::cout << "(>^ω^<)喵" << std::endl;
//	}
//};
//void letsHear(const Animal& animal)
//{
//	animal.talk();
//}
//int main()
//{
//	Cat cat;
//	Dog dog;
//	letsHear(cat);
//	letsHear(dog);
//	return 0;
//}//class A {};
//class B : public A {};
//class Person {
//public:
//	virtual A* BuyTicket()
//	{
//		cout << "买票-全价" << endl;
//		return nullptr;
//	}
//};
//class Student : public Person {
//public:
//	virtual B* BuyTicket()
//	{
//		cout << "买票-打折" << endl;
//		return nullptr;
//	}
//};
//void Func(Person* ptr)
//{
//	ptr->BuyTicket();
//}
//int main()
//{
//	Person ps;
//	Student st;
//	Func(&ps);
//	Func(&st);
//	return 0;
//}//class A
//{
//public:
//	virtual ~A()
//	{
//		cout << "~A()" << endl;
//	}
//};
//class B : public A {
//public:
//	~B()
//	{
//		cout << "~B()->delete:" << _p << endl;
//		delete _p;
//	}
//protected:
//	int* _p = new int[10];
//};
//// 只有派⽣类的析构函数重写了基类的析构函数,下⾯的delete对象调⽤析构函数,才能
//// 构成多态,才能保证p1和p2指向的对象正确的调⽤析构函数。
//int main()
//{
//	A* p1 = new A;
//	A* p2 = new B;
//	delete p1;
//	delete p2;
//	return 0;
//}// error C3668: “Benz::Drive”: 包含重写说明符“override”的⽅法没有重写任何基类⽅法
//class Car {
//public:
//	virtual void Dirve()
//	{}
//};
//class Benz :public Car {
//public:
//	virtual void Drive() override { cout << "Benz-舒适" << endl; }
//};
//int main()
//{
//	return 0;
//}// error C3248: “Car::Drive”: 声明为“final”的函数⽆法被“Benz::Drive”重写
//class Car
//{
//public:
//	virtual void Drive() final {}
//};
//class Benz :public Car
//{
//public:
//	virtual void Drive() { cout << "Benz-舒适" << endl; }
//};
//int main()
//{
//	return 0;
//}//class Car
//{
//public:
//	virtual void Drive() = 0;
//};
//class Benz :public Car
//{
//public:
//	virtual void Drive()
//	{
//		cout << "Benz-舒适" << endl;
//	}
//};
//class BMW :public Car
//{
//public:
//	virtual void Drive()
//	{
//		cout << "BMW-操控" << endl;
//	}
//};
//int main()
//{
//	// 编译报错:error C2259: “Car”: ⽆法实例化抽象类
//	// Car car;
//	Car* pBenz = new Benz;
//	pBenz->Drive();
//	Car* pBMW = new BMW;
//	pBMW->Drive();
//	return 0;
//}//class Person {
//public:
//	virtual void BuyTicket() { cout << "买票-全价" << endl; }
//private:
//	string _name;
//};
//class Student : public Person {
//public:
//	virtual void BuyTicket() { cout << "买票-打折" << endl; }
//private:
//	string _id;
//};
//class Soldier : public Person {
//public:
//	virtual void BuyTicket() { cout << "买票-优先" << endl; }
//private:
//	string _codename;
//};
//void Func(Person* ptr)
//{
//	// 这⾥可以看到虽然都是Person指针Ptr在调⽤BuyTicket
//	// 但是跟ptr没关系,⽽是由ptr指向的对象决定的。
//	ptr->BuyTicket();
//}
//int main()
//{
//	// 其次多态不仅仅发⽣在派⽣类对象之间,多个派⽣类继承基类,重写虚函数后
//	// 多态也会发⽣在多个派⽣类之间。
//	Person ps;
//	Student st;
//	Soldier sr;
//	Func(&ps);
//	Func(&st);
//	Func(&sr);
//}//// ptr是指针+BuyTicket是虚函数满⾜多态条件。
//// 这⾥就是动态绑定,编译在运⾏时到ptr指向对象的虚函数表中确定调⽤函数地址
//ptr->BuyTicket();
//00EF2001 mov eax, dword ptr[ptr]
//00EF2004 mov edx, dword ptr[eax]
//00EF2006 mov esi, esp
//00EF2008 mov ecx, dword ptr[ptr]
//00EF200B mov eax, dword ptr[edx]
//00EF200D call eax
//// BuyTicket不是虚函数,不满⾜多态条件。
//// 这⾥就是静态绑定,编译器直接确定调⽤函数地址
//ptr->BuyTicket();
//00EA2C91 mov ecx, dword ptr[ptr]
//00EA2C94 call Student::Student(0EA153Ch)//class Base {
//public:
//	virtual void func1() { cout << "Base::func1" << endl; }
//	virtual void func2() { cout << "Base::func2" << endl; }
//	void func5() { cout << "Base::func5" << endl; }
//protected:
//	int a = 1;
//};
//class Derive : public Base
//{
//public:
//	// 重写基类的func1
//	virtual void func1() { cout << "Derive::func1" << endl; }
//	virtual void func3() { cout << "Derive::func1" << endl; }
//	void func4() { cout << "Derive::func4" << endl; }
//protected:
//	int b = 2;
//};
//int main()
//{
//	Base b;
//	Derive d;
//	return 0;
//}//int main()
//{
//	int i = 0;
//	static int j = 1;
//	int* p1 = new int;
//	const char* p2 = "xxxxxxxx";
//	printf("栈:%p\n", &i);
//	printf("静态区:%p\n", &j);
//	printf("堆:%p\n", p1);
//	printf("常量区:%p\n", p2);
//	Base b;
//	Derive d;
//	Base* p3 = &b;
//	Derive* p4 = &d;
//	printf("Person虚表地址:%p\n", *(int*)p3);
//	printf("Student虚表地址:%p\n", *(int*)p4);
//	printf("虚函数地址:%p\n", &Base::func1);
//	printf("普通函数地址:%p\n", &Base::func5);
//	return 0;
//}

http://www.dtcms.com/a/314812.html

相关文章:

  • 强反射场景识别误差↓78%!陌讯多模态融合算法在水位监测的落地优化
  • Shell操作git,上传更新文档
  • Redshift 渲染器:GPU 加速渲染的高效之选
  • TGD第十一篇:卷积神经网络中的TGD特征
  • MS-DOS 常用指令集
  • OCR 精准识别验讫章:让登记与校验更智能
  • ssh连接VirtualBox中的Ubuntu24.04(win11、putty、NAT 模式)
  • 西门子PLC S7-1200单轴步进控制电动机
  • Exporters | 安装process_exporter
  • C语言:构造类型学习
  • 深入剖析Java Stream API性能优化实践指南
  • 【Django】-11- 后台管理界面定制
  • [机器学习]02-基于贝叶斯决策的鸢尾花数据集分类
  • 云原生攻防6(Kubernetes扩展知识)
  • 并发编程常用工具类(下):CyclicBarrier 与 Phaser 的协同应用
  • 政府财政行业云原生转型之路
  • 关于解决WinRiver项目动态XmlElement的序列化与反序列化的问题
  • 基于Java的AI工具和框架
  • PyTorch生成式人工智能(25)——基于Transformer实现机器翻译
  • spring boot开发中的资源处理等问题
  • RTOS如何保证实时性
  • 深圳南柯电子|电驱动系统EMC测试整改:“诊断-治疗-预防”方案
  • HTML5的新特性
  • 上位机知识篇---令牌
  • 如何选择合适的政务文本检测工具?
  • go 语言常见问题(2)
  • 宝塔面板安装WordPress教程:10分钟一键部署搭建个人博客 (2025)
  • 聊聊web前端的缓存问题
  • 金融专业高分简历撰写指南
  • k8s集群