关于多态

本文详细介绍了C++中的多态概念,包括静态多态和动态多态,并通过示例代码展示了虚函数、纯虚函数及抽象类的作用。此外,还讨论了协变的概念及其与重写的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

多态基本概念

顾名思义就是多种形态,多种状态。

分类

这里写图片描述

静态多态

编译器在编译期间完成的,编译器根据函数实参的类型(可能会进行隐式类型的转换),可推断出要调用哪一个函数,如果有对应的函数就调用该函数,否则出现编译错误。

#include<iostream>
#include<stdlib.h>
using namespace std;
class C
{
public:
    int fun(int x,int y,int z)
    {
        cout<< x + y + z<<endl;
        return 0;
    }
    int fun(int s)
    {
        cout<<s<<endl;
        return 0;
    }
};
void test()
{
    C a;
    a.fun(1, 2, 3);
    a.fun(4);
}

这里写图片描述
可以看到,在C类里面两个fun()构成了重载,作用域函数名都一样,参数个数不一样,而在test()里面,通过C类的对象调用两个fun()的时候,可以通过参数决定调用哪个fun(),这里就实现了静态多态。

动态多态

1:首先了解什么叫做虚函数

class A
{
public:
     void funtest1()
    {
        cout << "A.funtest1" << endl;
    }
};
class B: public A
{
public:
    void funtest1()
    {
        cout << "B.funtest1" << endl;
    }
};
void fun(A &s)
{
    s.funtest1();
}
int main()
{
    A d;
    fun(d);
    B c;
    fun(c);
    system("pause");
    return 0;
}

这里写图片描述
在上面的代码里面,fun(A &s)的参数是基类的引用,而在主函数里面不论用基类的对象还是派生类的对象去调用fun函数时,所执行的都是基类中的成员函数,而我们期望的是当我们用派生类对象调用fun函数时会执行派生类的成员函数,这里就需要虚函数的帮助来实现。那么,什么是虚函数呢?

虚函数

在类的声明中被加上virtual关键字修饰的成员函数。虚函数可以实现动态多态,在派生类中进行重写。

#include<iostream>
#include<stdlib.h>
using namespace std;
class A
{
public:
     virtual void funtest1()
    {
        cout << "A.funtest1" << endl;
    }
};
class B: public A
{
public:
    virtual void funtest1()
    {
        cout << "B.funtest1" << endl;
    }
};
void fun(A &s)
{
    s.funtest1();
}
int main()
{
    A d;
    fun(d);
    B c;
    fun(c);
    system("pause");
    return 0;
}

这里写图片描述
可以看出,通过虚函数便可以实现用实际传参来决定执行哪一个类的成员函数,也就是动态多态。

动态多态具体概念

当一个函数如上例中func()函数那样以基类对象引用(或基类对象指针)做参数时,调用该函数时给的实际参数可以是基类的对象也可以是派生类的对象(或对象地址),这时希望在运行函数时根据实际参数的类型来决定是调用基类的还是派生类的成员函数(该函数在基类与派生类中原型相同),这种情况称为动态多态。

c++中如何实现多态

1:在继承体系下,基类必须有虚函数。
2:派生类必须对基类的虚函数进行重写。

继承体系中同名函数的关系

这里写图片描述
什么叫协变?
协变:与重写条件基本相同,唯一不同协变的返回值不同,基类返回基类指针或引用,派生类返回派生类的指针或引用。

纯虚函数

在成员函数的形参后面写上=0.则成员函数为纯虚函数。

#include<iostream>
#include<stdlib.h>
using namespace std;
class A    //抽象类
{
public:
    virtual void funtest1() = 0   //纯虚函数
    {
        cout << "A.funtest1" << endl;
    }
};
class B: public A
{
public:
    virtual void funtest1()
    {
        cout << "B.funtest1" << endl;
    }
};
void fun(A &s)
{
    s.funtest1();
}
int main()
{
    A d;
    fun(d);
    B c;
    fun(c);
    system("pause");
    return 0;
}

这里写图片描述
对上面这段代码进行变异发现编译器会报如上图所表示的错误,那么就又有一个概念叫做抽象类,

抽象类

含有纯虚函数的类叫做抽象类,抽象类是不能实例出对象的。纯虚函数在派生类中重新定义以后,派生类才能实例出对象。

#include<iostream>
#include<stdlib.h>
using namespace std;
class A    //抽象类
{
public:
    virtual void funtest1() = 0   //纯虚函数
    {
        cout << "A.funtest1" << endl;
    }
};
class B: public A
{
public:
    virtual void funtest1()
    {
        cout << "B.funtest1" << endl;
    }
};
void fun(A &s)
{
    s.funtest1();
}
int main()
{
    B c;
    fun(c);
    system("pause");
    return 0;
}

可以看到在派生类B中对纯虚函数重写以后是可以实例出对象的,通过编译完全没问题。
总结:
1、派生类重写基类的虚函数实现多态,要求函数名、参数列表、返回值完全相同。(协变除外)
2、基类中定义了虚函数,在派生类中该函数始终保持虚函数的特性。
3、只有类的非静态成员函数才能定义为虚函数,静态成员函数不能定义为虚函数。
4、如果在类外定义虚函数,只能在声明函数时加virtual关键字,定义时不用加。
5、构造函数不能定义为虚函数,虽然可以将operator=定义为虚函数,但最好不要这么做,使用时容易混淆6、不要在构造函数和析构函数中调用虚函数,在构造函数和析构函数中,对象是不完整的,可能会出现未定义的行为。7、最好将基类的析构函数声明为虚函数。(析构函数比较特殊,因为派生类的析构函数跟基类的析构函数名称不一样,但是构成覆盖,这里编译器做了特殊处理)。
8、虚表是所有类对象实例共用的。

### C++ 中的多态示例 #### 示例 1:基于虚函数的动态多态 下面是一个完整的 C++ 示例,展示了如何利用虚函数实现动态多态: ```cpp #include <iostream> using namespace std; // 基类 Animal class Animal { public: virtual void sound() { // 虚函数 cout << "This is an animal." << endl; } }; // 派生类 Dog class Dog : public Animal { public: void sound() override { // 重写基类的虚函数 cout << "Dog says: Woof!" << endl; } }; // 派生类 Cat class Cat : public Animal { public: void sound() override { // 重写基类的虚函数 cout << "Cat says: Meow!" << endl; } }; // 函数接受基类引用作为参数 void makeSound(Animal& animal) { animal.sound(); // 根据实际对象类型调用对应的虚函数 } int main() { Dog d; Cat c; makeSound(d); // 输出 "Dog says: Woof!" makeSound(c); // 输出 "Cat says: Meow!" return 0; } ``` 此代码展示了一个典型的动态多态场景。`makeSound` 函数接收 `Animal` 类型的引用,但在运行时会根据实际对象类型调用相应的 `sound()` 方法[^1]。 --- #### 示例 2:静态多态(函数重载) 以下是静态多态的一个简单例子,涉及函数重载: ```cpp #include <iostream> using namespace std; // 不同类型的参数触发不同版本的函数 void display(int value) { cout << "Integer Value: " << value << endl; } void display(double value) { cout << "Double Value: " << value << endl; } int main() { display(42); // 调用整数版本 display(3.14); // 调用双精度浮点数版本 return 0; } ``` 在这个例子中,编译器会在编译阶段决定调用哪个具体的 `display` 函数,这属于静态多态的一部分[^3]。 --- #### Python 中的多态示例 尽管问题是针对 C++ 的,但为了对比说明,提供一个简单的 Python 多态示例: ```python # 定义基类 class Bird: def fly(self): print("Bird can fly.") # 定义派生类 class Sparrow(Bird): def fly(self): print("Sparrow flies low.") class Eagle(Bird): def fly(self): print("Eagle soars high.") def bird_fly(bird_obj): bird_obj.fly() sparrow = Sparrow() eagle = Eagle() bird_fly(sparrow) # 输出 "Sparrow flies low." bird_fly(eagle) # 输出 "Eagle soars high." ``` 这段代码演示了 Python 如何通过继承和方法覆盖来实现多态行为[^4]。 --- ### 关键知识点总结 - **动态多态**依赖于虚函数机制,在运行时确定具体调用的方法。 - **静态多态**通常由函数重载或运算符重载实现,其绑定发生在编译期间。 - 多态的核心在于“父类引用/指针指向子类对象”,并通过调用被重写的成员函数展现差异化的功能[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值