C++中的多态性

定义:

子类重新定义父类中有相同名称,返回值和参数的虚函数,主要在继承关系中出现。

基本条件:

重写的函数和被重写的函数必须都为virtual函数,并分别位于基类和派生类中,如果父类被重写的函数加了

vritual,子类继承父类,那么子类重写的函数加不加vritual都是虚函数

重写的函数和被重写的函数,返回值,函数名和函数参数必须完全一致;

cpp 复制代码
#include<iostream>
    using namespace std;
    class father
    {
    public:
    	virtual void fun()
    	{
    		cout << "father" << endl;
    	}
    };
    class child:public father
    {
    public:
    	virtual void fun()
    	{
    		cout << "child" << endl;
    	}
    };

函数隐藏

定义:在子类中只要和父类函数名字相同不是重写,一定是函数隐藏。

cpp 复制代码
class Base
    {
    	public:
    	void funA(){cout<<"funA()"<<endl;} 
    	virtual void funB(){cout<<"funB()"<<endl;} 
    };
    class Heri:public Base
    {
    	public:
    	void funA(){cout<<"funA():Heri"<<endl;} //函数隐藏因为不是虚函数
    	void funA(int a){cout<<"funA(int a):heri"<<a<<endl;} //函数隐藏参数不同
    	void funB(){cout<<"funB():heri"<<endl;} //函数重写
    };

多态的基本概念

静态多态: 函数重载 和 运算符重载属于静态多态,复用函数名

动态多态: 派生类和虚函数实现运行时多态

静态多态和动态多态区别:

静态多态的函数地址早绑定 - 编译阶段确定函数地址

动态多态的函数地址晚绑定 - 运行阶段确定函数地址

多态的基础: 需要有重写,子类重写父类的 返回值 名字 参数相同的虚函数,(只要父类的函数是虚函数即可)。

动态多态: 父类的指针或引用指向子类对象,并且通过该指针或引用调用子类重写的虚函数

多态的使用条件: 父类的指针或引用指向子类对象

cpp 复制代码
#include<iostream>
    #include<string>
    #include<vector>
    using namespace std;
    class Animal
    {
    public:
    	virtual void speak()
    	{
    		cout << "animal speak" << endl;
    	}
    	void work()
    	{
    		cout << "animal work" << endl;
    	}
    };
    class cat :public Animal
    {
    public:
    	void speak()
    	{
    		cout << "cat speak" << endl;
    	}
    	void work()
    	{
    		cout << "cat work" << endl;
    	}
    };
    class dog :public Animal
    {
    public:
    	void speak()
    	{
    		cout << "dog speak" << endl;
    	}
    	void work()
    	{
    		cout << "dog work" << endl;
    	}
    };
    int main() {
    	
    	Animal *a = new dog;
    	//speak 是函数重写 ,work是函数隐藏
    	a->speak();//调用的是dog里的speak函数
    	a->work();//调用的函数是隐藏函数时,调用函数的指针或对象是什么类型调用哪里的函数,调用的是animal里的work函数
    	return 0;
    }

C++中如何实现多态:

类的虚函数表

每个包含了虚函数的类都包含一个虚函数表(存放虚函数指针的数组,就是存放的是虚函数的地址),调用虚函

数,必须通过虚函数表找到虚函数的地址

当一个类(B)继承另一个类(A)时,类 B 会继承类 A 的函数的调用权。所以如果一个基类包含了虚函数,那么

其继承类也可调用这些虚函数,换句话说,一个类继承了包含虚函数的基类,那么这个类也拥有自己的虚函数表。

虚函数表在编译阶段被初始化

cpp 复制代码
class A {
    public:
    	virtual void vfunc1();
    	virtual void vfunc2();
    	void func1();
    	void func2();
    private:
    	int m_data1, m_data2;
    };
    class B : public A{//此时类B也拥有自己的虚表
    };

虚表指针

只要类中有虚函数,那么类中就会多一个成员变量,这个成员变量就是虚函数表的指针,调用虚函数,先访问这个对象里的虚函数表指针,通过虚函数表指针找到虚函数那个数组,在遍历数组找到虚函数地址

cpp 复制代码
class A {
    public:
        virtual void vfunc1();
        virtual void vfunc2();
        void func1();
        void func2();
    private:
        int m_data1, m_data2;
    };
    
    class B : public A {
    public:
        virtual void vfunc1();
        void func1();
    private:
        int m_data3;
    };
    
    class C: public B {
    public:
        virtual void vfunc2();
        void func2();
    private:
        int m_data1, m_data4;
    };
    
    int main() 
    {
        B bObject;
        A *p = & bObject;
        p->vfunc1();//父类指针调用虚函数,先通过父类指针访问对象bObject,找到对象里的虚表指针,这个虚表指针指向的虚表是类B的虚表,就上类B的虚表中找vfun1,类B重写了vfun1,所以调用的是子类的vfun1
    }

多态的好处:

1.更好的代码重用性:多态允许你使用父类指针或引用来操作子类对象,这意味着你可以编写通用的代码来处理不同类型的对象,从而提高代码的重用性。

2.代码的可扩展性:多态使得代码更加灵活和可扩展。当你需要添加新的子类时,不需要修改现有的代码,只需要在新子类中实现相应的方法即可。

3.更好的灵活性:多态使得你能够根据具体情况选择不同的实现方式,而不需要拘泥于固定的实现。这在处理复杂问题时非常有用。

相关推荐
Wish3D31 分钟前
阿里云OSS 上传文件 Python版本
开发语言·python·阿里云
凤年徐32 分钟前
【数据结构初阶】单链表
c语言·开发语言·数据结构·c++·经验分享·笔记·链表
oioihoii34 分钟前
C++11 右值引用:从入门到精通
开发语言·c++
朝新_4 小时前
【多线程初阶】阻塞队列 & 生产者消费者模型
java·开发语言·javaee
立莹Sir4 小时前
Calendar类日期设置进位问题
java·开发语言
木子.李3474 小时前
排序算法总结(C++)
c++·算法·排序算法
风逸hhh5 小时前
python打卡day46@浙大疏锦行
开发语言·python
火兮明兮5 小时前
Python训练第四十三天
开发语言·python
freyazzr6 小时前
C++八股 | Day2 | atom/函数指针/指针函数/struct、Class/静态局部变量、局部变量、全局变量/强制类型转换
c++
ascarl20106 小时前
准确--k8s cgroup问题排查
java·开发语言