首页 > 编程语言 >深入挖掘C++中的特性之一 — 多态

深入挖掘C++中的特性之一 — 多态

时间:2024-10-08 12:47:16浏览次数:10  
标签:函数 多态 C++ class virtual 挖掘 重写 public

目录

1.多态的概念

2.多态的定义及其实现

1.虚函数

2.虚函数的重写/覆盖

3.实现多态的必要条件

4.多态的代码呈现

5.来一道小题,深入理解一下多态

3.虚函数重写的一些其他问题

1.协变

2.析构函数的重写

4.override和final关键字

5.重载/重写/隐藏的对比(相同函数名的函数间关系)

6.纯虚函数和抽象类

7.多态的原理

1.虚函数表指针

1.引入:

2.介绍

2.多态是如何实现的

3.动态绑定与静态绑定

4.虚函数表


1.多态的概念

2.多态的定义及其实现

1.虚函数

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

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

2.虚函数的重写/覆盖

3.实现多态的必要条件

1.必须是基类的指针或引用调用虚函数。

2.派⽣类必须对基类的虚函数重写/覆盖,重写或者覆盖了,派⽣类才能有不同的函数,多态的不同形态效果才能达到。

注意:在多态的场景下,基类指针或引用去调用哪个函数,已经不是看它是父类的指针或引用就把它归结于去调用父类的函数去了,调用的是子类的函数还是父类的函数应该看的是这个父类的指针或引用指向的是父类的对象还是子类的对象;指向父类我调用的就是父类的那个函数,指向子类我调用的就是子类的那个函数。

4.多态的代码呈现

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;
}

结果:

5.来一道小题,深入理解一下多态

class A
{
public:
	virtual void func(int val = 1) { std::cout << "A->" << val << std::endl; }
	virtual void test() { func(); }
};

class B : public A
{
public:
	void func(int val = 0) { std::cout << "B->" << val << std::endl; }
};

int main(int argc, char* argv[])
{
	B* p = new B;
	p->test();
	return 0;
}

这道题的运行结果是什么?

结果:

分析过程:

注意:

多态中对于虚函数的重写,本质上是对虚函数的实现进行了重写。

3.虚函数重写的一些其他问题

1.协变

举个例子:

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;
}

2.析构函数的重写

4.override和final关键字

从上⾯可以看出,C++对虚函数重写的要求⽐较严格,但是有些情况下由于疏忽,⽐如函数名写错参数写错等导致⽆法构成重写,⽽这种错误在编译期间是不会报出的,只有在程序运行时没有得到预期结果才来debug会得不偿失,因此C++11提供了override,可以帮助用户检测是否重写。如果我们不想让派⽣类重写这个虚函数,那么可以⽤final去修饰。

5.重载/重写/隐藏的对比(相同函数名的函数间关系)

6.纯虚函数和抽象类

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;
}

7.多态的原理

1.虚函数表指针

1.引入:

class Base
{
public:
	virtual void Func1()
	{
		cout << "Func1()" << endl;
	}
protected:
	int _b = 1;
	char _ch = 'x';
};


int main()
{
	Base b;
	cout << sizeof(b) << endl;
	return 0;
}

x86环境下的结果:

出现这样结果的原因就是这个类创建的对象中存放了虚函数表指针。

2.介绍

监视窗口下的b变量

这里的_vfptr就是虚函数表指针。

⼀个含有虚函数的类中都至少都有⼀个虚函数表指针,因为⼀个类所有虚函数的地址要被放到这个类对象的虚函数表中,虚函数表也简称虚表。

2.多态是如何实现的

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

class Student : public Person {
public:
	virtual void BuyTicket() { cout << "买票-打折" << endl; }

protected:
	int _id;
};

class Soldier : public Person {
public:
	virtual void BuyTicket() { cout << "买票-优先" << endl; }

protected:
	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);

	return 0;
}

_vfptr可以认为是从基类继承(直接复制)到派生类来的,但只要在派生类中实现了重写一部分虚函数,那么就会将原有的这一部分虚函数覆盖。

3.动态绑定与静态绑定

4.虚函数表

标签:函数,多态,C++,class,virtual,挖掘,重写,public
From: https://blog.csdn.net/GGbond665/article/details/142716993

相关文章

  • 【算法】博弈论(C/C++)
    个人主页:摆烂小白敲代码创作领域:算法、C/C++持续更新算法领域的文章,让博主在您的算法之路上祝您一臂之力欢迎各位大佬莅临我的博客,您的关注、点赞、收藏、评论是我持续创作最大的动力目录博弈论:1.Grundy数与Nim博弈Nim博弈规则:Grundy数的计算:例题:2.极大极小算法......
  • 每天学点C++之多态
    一多态的语法多态的基本含义是,函数参数父为类型,传入该类不同的子类,所表现的行为会根据子类的具体情况而不同,大家请看代码:#include<iostream>usingnamespacestd;classAnimal{public:voideat(){cout<<"Animaleat"<<endl;}};classCat:publ......
  • 【C++】速通涉及 “vector” 的经典OJ编程题
    【C++】速通涉及“vector”的经典OJ编程题一.杨辉三角解题思路:代码实现:二.删除有序数组中的重复项解题思路:代码实现:【C/C++】按位运算符使用规制三.只出现一次的数字解题思路:代码实现:四.只出现一次的数字III解题思路:代码实现:一.杨辉三角本题LeetCode链......
  • SRC漏洞挖掘----信息搜集
    信息搜集插件推荐:shodan,findsomething,retire.js,hacktool,FoFaProviewSRC漏洞挖掘之信息搜集资产搜集的网站:zoomeye,fofa(收费),360网络空间测绘系统-360数字安全(免费)1.FOFA语法学习介绍:FOFA(FingerprintingOrganizationswithAdvancedTools)是一家总部位于中国的网络安全公......
  • c++条件变量
    条件变量是用于线程间同步的一种机制,它允许一个或多个线程在某个条件满足之前等待,并在条件满足时通知等待的线程继续执行。以下是条件变量的基本使用方法,包括notify_one和notify_all的作用。使用条件变量的基本步骤创建条件变量和互斥量:首先需要创建一个std::condition_v......
  • Java基础第八章(多态)
    多态1.方法体现多态方法重载体现多态对象通过传入不同数量的参数,会调用不同的sun方法,体现出多态方法重写体现多态A类和B类是继承关系,通过不同对象调用对应类中重写的方法体现2.对象体现多态编译是javac,运行是java(1)一个对象的编译类型和运行类型可以不一致将父......
  • c++指针传递与引用传递
    c不支持引用传递的!在C++中,指针传递和引用传递是两种常用的参数传递方式,它们各自有不同的特点和适用场景。下面是两者之间的主要区别:1.语法和使用指针传递定义和调用:函数参数是一个指针类型,调用时需要传递变量的地址。解引用:在函数内部需要使用解引用操作符*来访问指针......
  • c++可变模板参数
    在C++中的可变模板参数使用省略号...来表示一个参数包(ParameterPack),其具体位置决定了这个包是模板参数包还是函数参数包,以及如何进行参数展开。1.模板参数包:c...Args省略号放在类型名称的右边,用来表示模板参数包,即可以接受任意数量的模板类型参数。template<typename...A......
  • java_day8_多态、抽象类、接口
    一、多态多态的好处:1、提高了程序的维护性(由继承保证)2、提高了程序的扩展性(由多态保证)代码案例(多态的拓展性)classAnimal{publicvoideat(){System.out.println("吃");}publicvoidsleep(){System.out.println("睡");}}clas......
  • C++ 对象模型
    对象的内存布局非虚函数类对象对于不包含虚函数的类,对象的内存布局相对简单,其成员变量按照声明的顺序依次存储。例如:classSimpleClass{private:intnum;doubled;public:SimpleClass(intn,doubledd):num(n),d(dd){}};在SimpleClass对象的内存......