c++的多态和虚函数

什么是多态

直观的说,就是向不同的对象发送一个消息,不同的对象在接收消息以后有不同的行为;函数重载和运算符重载都是多态现象;
多态分为静态多态和动态多态;
静态多态是通过函数重载实现的,比如收函数重载和运算符重载;静态重载要求程序编译的时候就知道调用函数的信息,在编译的时候就要知道调用那个函数;静态多态又称为编译时的多态性;
动态多态是在运行过程中才动态的确定操作所针对的对象,又称为运行时的多态性;动态多态性是由虚函数实现的

动态多态性研究的问题:当一个基类被继承为不同的派生类时,各派生类可以使用与基类成员相同的成员名,如果的时候用同一个成员名调用类对象的成员,会调用那个对象的成员呐?

虚函数在多态中的运用

这点我们从上一篇文章的最后一段话引入;派生类对象可以给基类对象赋值,基类对象的指针也可以指向派生类的对象,但是基类对象指针指向派生类以后就只能指向派生类中从基类继承归过来的那一部分内容;首先让我们看看这一点有多不方便

  • 例子1
#include<iostream>
#include<vector>
#include<queue>
using namespace std;
class student
{
public:
	string name;
	int score;
	student(string n, int s)
	{
		name = n;
		score = s;
	}
	void display()
	{
		cout << "name:" << name << endl;
		cout << "score:" << score << endl;
		return;
	}
};

class graduate:public student
{
public:
	int wage;
	graduate(string s, int n, int w) :student(s,n)
	{
		wage = w;
	}
	void dispaly()
	{
		cout << "graduate's name is" << name << endl;
		cout << "graduate's score is" << score << endl;
		cout << "graduate's wage is" << wage << endl;
		return;
	}
};
int main()
{
	student stu("feng haojun", 100);
	graduate gra("haojun", 100, 100000);
	student* pt = &stu;
	pt->display();
	pt = &gra;
	cout << endl;
	pt->display();
	return 0;
}
name:feng haojun
score:100

name:haojun
score:100

可见使用同一个指针,当基类指针指向派生类对象的时候,指针只能访问派生类从基类继承过去的一部分成员;并不能访问派生类中的另加成员;这里指向派生类的指针调用的还是基类中的display函数,在派生类的构造函数中对基类的变量进行了初始化,然后使用基类的display函数就调用了基类中被派生类构造函数重新初始化以后的变量;意思就是说指向基类的指针始终还是指向基类的;下边就引出解决办法,也就是虚函数的大作用的体现,也是c++多态性的直观体现

  • 例子2
#include<iostream>
#include<vector>
#include<queue>
using namespace std;
class student
{
public:
	string name;
	int score;
	student(string n, int s)
	{
		name = n;
		score = s;
	}
	virtual void display();
};
void student::display()
{
	cout << "name:" << name << endl;
	cout << "score:" << score << endl;
	return;
}

class graduate:public student
{
public:
	int wage;
	graduate(string s, int n, int w) :student(s,n)
	{
		wage = w;
	}
	void display();
};

void graduate::display()
{
	cout << "graduate's name is:" << name << endl;
	cout << "graduate's score is:" << score << endl;
	cout << "graduate's wage is:" << wage << endl;
	return;
}

int main()
{
	student stu("feng haojun", 100);
	graduate gra("haojun", 100, 100000);
	student* pt = &stu;
	pt->display();
	pt = &gra;
	cout << endl;
	pt->display();
	return 0;
}
name:feng haojun
score:100

graduate's name is:haojun
graduate's score is:100
graduate's wage is:100000

基类中的函数使用虚函数,子类的虚函数在派生类中可以被改写;在声明派生类时被重载,这时派生类的同名函数diapaly就取代了基类中的虚函数,所以使用基类的指针指向派生类的对象以后,调用的display函数就调用了派生类的diaplay函数;

由虚函数实现的动态多态性就是:同一类族中不同类的对象,随同一函数调用函数作出不同的响应

一个概念辨析:
在基类中定义的非虚函数会在派生类中被重新定义,如果使用基类指针调用该成员函数,则系统会调用对象中基类部分的成员函数;如果派生类指针调用该成员函数,则系统会调用派生类对象中的成员函数;这不是多态性;这单纯就是使用不同类型的指针实现的功能,没有使用道虚函数的功能

静态关联和动态关联

在梳理这部分的知识以前,先来收一收重写和重载的区别
重载和重写的区别
(1)范围区别:重写和被重写的函数在不同的类中,重载和被重载的函数在同一类中。
(2)参数区别:重写与被重写的函数参数列表一定相同,重载和被重载的函数参数列表一定不同。
(3)virtual的区别:重写的基类函数必须要有virtual修饰,重载函数和被重载函数可以被virtual修饰,也可以没有

静态关联:函数重载和通过对象名调用的虚函数,在编译时就确定了调用的虚函数属于哪一类,这个过程叫静态关联;也叫早期关联

动态关联
动态关联是运行期的操作,首先定义一个指向基类的指针变量,并使其指向相应的类变量,然后使用这个指针去调用某虚函数,这样的调用方式,编译系统在编译的时候是无法确定调用哪一类对象的虚函数的;在运行阶段,该指针首先指向某一个基类对象,调用基类对象中某虚成员函数;然后指针指向了一个派生类,使用派生类指向派生类中与虚函数同名的函数,这个时候就完成了虚函数和类对象的绑定;就叫做动态关联

什么情况下声明虚函数

虚函数使用的场合:

  1. 成员函数所在的类如果会作为基类,就可以将该成员函数声明为虚函数
    2)如果成员函数在继承以后无需修改,或者派生类中用不上该函数,就不需要声明为虚函数
    3) 成员函数的调用是通过对象名还是通过基类指针调用;如果是使用基类指针调用的话就需要将该成员函数定义为虚函数
    4) 基类中不实现该成员函数,函数体为空;具体实现留给派生类去添加;需要将该成员函数声明为虚寒苏

虚折构函数

为什么要使用虚析构函数:
直接的讲,C++中基类采用virtual虚析构函数是为了防止内存泄漏。具体地说,如果派生类中申请了内存空间,并在其析构函数中对这些内存空间进行释放。假设基类中采用的是非虚析构函数,当删除基类指针指向的派生类对象时就不会触发动态绑定,因而只会调用基类的析构函数,而不会调用派生类的析构函数。那么在这种情况下,派生类中申请的空间就得不到释放从而产生内存泄漏。所以,为了防止这种情况的发生,C++中基类的析构函数应采用virtual虚析构函数

纯虚函数

在基类中将某成员函数定义成为虚函数,并不是基类需要这个函数,而是考虑到派生类的需要,只是在基类中预留了一个函数名,具体功能由派生类实现;给出函数的原型,并在后边加上”==0“

如果在基类中没有保留函数名字,那么在派生类中就无法实现多态
包含纯虚函数的类是抽象类,无法建立对象;抽象类只是让派生类去继承的;
如果派生类中没有对所有的纯虚函数进行定义,则此派生类仍然是抽象类,不能用于定义对象;

小记:输入输出运算符重载只能在类外,通过友函数的形式在类内声明;不能直接在类内定义

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
C++中的继承、多态虚函数是面向对象编程的重要概念。 继承是指一个类可以从另一个类继承属性和方法。子类可以继承父类的公有成员和保护成员,但不能继承私有成员。通过继承,子类可以重用父类的代码,并且可以添加自己的特定功能。继承可以实现代码的重用和层次化的设计。 多态是指同一个函数可以根据不同的对象调用不同的实现。多态可以通过虚函数来实现。虚函数是在基类中声明为虚拟的函数,它可以在派生类中被重写。当通过基类指针或引用调用虚函数时,实际调用的是派生类中的实现。这样可以实现动态绑定,即在运行时确定调用的函数。 虚函数的原理是通过虚函数表来实现的。每个包含虚函数的类都有一个虚函数表,其中存储了虚函数的地址。当调用虚函数时,编译器会根据对象的类型在虚函数表中查找对应的函数地址并调用。 综上所述,C++中的继承、多态虚函数是实现面向对象编程的重要机制,它们可以提高代码的灵活性和可扩展性。 #### 引用[.reference_title] - *1* *3* [C++多态虚函数虚函数表](https://blog.csdn.net/weixin_46053588/article/details/121231465)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [c++多态虚函数表内部原理实战详解](https://blog.csdn.net/bitcarmanlee/article/details/124830241)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

星光技术人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值