C++的多态

C++三大特性:封装、继承和多态。其中最好理解的就是封装了,继承作为C++面向对象的特征也不难理解,那么多态,应该也不太难。本文来讨论一下多态是什么,有什么特殊之处。

1、什么是多态

多态,就是说多种形态。

多态分为静态多态和动态多态两部分:

  • 静态多态:在编译前就确定了调用的具体函数,也就是重载。(本文不讨论,请参见C++的重载)
  • 动态多态:运行时确认调用的具体函数,也就是我们一般叫的多态。

我们只具体讨论一下多态(动态多态)。
多态是在不同继承关系的类对象,去调同一函数,产生了不同的行为。就是说,有一对继承关系的两个类,这两个类里面都有一个函数且名字、参数、返回值均相同,然后我们通过调用函数来实现不同类对象完成不同的事件。

2、多态的要素

多态有两个要素:

  • 要有继承关系的两个类,存在同名、同参数、同返回值的函数;
  • 拥有虚函数,且完成虚函数的重写。

3、多态的实现

上代码:

#include <iostream>

using namespace std;

class A
{
public:
    virtual void print()
    {
        cout << "A::print" << endl;
    }
};

class B : public A
{
public:
    virtual void print()
    {
        cout << "B::print" << endl;
    }
};

int main(int argc, char **argv)
{
    A a;
    B b;
    A *pa = &a;
    pa->print();

    pa = &b;
    pa->print();
    return 0;
}

//输出:
//A::print
//B::print

说白了就是这么个东西,可能不太好理解,那么我们尝试改改main的写法:

void Print(A &x)
{
    x.print();
}

int main(int argc, char **argv)
{
    A a;
    B b;
    Print(a);
    Print(b);
    return 0;
}
//输出:
//A::print
//B::print

这样就很明了了,其实和上面那种写法是一样的效果。要注意:

只能是子类给父类赋值,所以Print形参只能是父类类型。

4、多态的原理

在类中使用virtual声明虚函数时,编译器会在类中创建出一个虚函数指针指向创建的存放虚函数地址的虚函数表,虚函数表其实就是一个存放函数指针的数组。
在调用时候首先获取对象的首地址,然后解引用取到虚函数表的首地址,加上偏移量找到对应的虚函数,进行调用。

5、其他

5.1 为什么要用虚函数表?

  • 实现多态,父类对象调用父类的虚函数,子类对象调用的是子类的虚函数。
  • 同一个类的多个对象使用同一份虚函数表,可以节省空间占用;一个类自身的虚函数和继承的虚函数以及重写父类的虚函数都会存在自己的虚函数表。

5.2 基类虚构函数为什么要定义为虚函数?

使用基类操作派生类时候,为了防止只执行了基类的析构函数,没有执行派生类的析构函数,只能删除基类的对象,派生类的对象没有被删除,造成内存泄漏。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 第一次“扫人”,虽然资源量不多,但始终在一步一步增多。明天休息,好好休息,养精蓄锐。为后天做准备。 加油!!!
    欢欢1020阅读 1,241评论 0 0
  • 我以一个观察者的身份,来说说我看到的“移民”,特别是华人移民,所面临的尴尬情境。 以下是分享的Slides.
    大鱼_BigFish阅读 1,553评论 1 1
  • 放学了,小红,小芳和小明相约在公园里玩耍,小明比较调皮,他爬上了树,小红和小芳跑到草地上玩皮球。小美从路边经过,看...
    毕子辰阅读 1,618评论 0 1
  • 最近一直在想代码那些事!竟然昨晚还梦见敲代码,这也是醉了。梦到在终端写sql语句,突然断电,终端不能保存,好后悔没...
    书谨阅读 1,612评论 0 0
  • 今天晚上我写完作业,画了一副海绵宝宝的画,虽然我自己不太满意,妈妈还说挺漂亮的。然后我犹犹豫豫的告诉妈妈,这幅画其...
    为为的小世界阅读 3,064评论 0 5