美文网首页
C++ 之 多态(非常非常重要,重点在后面)

C++ 之 多态(非常非常重要,重点在后面)

作者: 程序爱好者 | 来源:发表于2019-08-03 16:06 被阅读0次

编译环境:WIN10 VS2017

这篇文章有点长,但都是满满的干货,一定要看到最后,那才是重点。

什么是多态?

顾名思义就是同一个事物在不同场景下的多种形态。

下面会具体的详细的介绍。

静态多态

我们以前说过的函数重载就是一个简单的静态多态

C/C++学习交流QQ群728483370,大家一起加油!

int Add(int left, int right)

{

    return left + right;

}

double Add(double left, int right)

{

    return left + right;

}

int main()

{

    Add(10, 20);

    //Add(10.0, 20.0);  //这是一个问题代码

    Add(10.0,20);  //正常代码

    return 0;

}

可以看出来,静态多态是编译器在编译期间完成的,编译器会根据实参类型来选择调用合适的函数,如果有合适的函数可以调用就调,没有的话就会发出警告或者报错。。。比较简单,不做多介绍。

动态多态

什么是动态多态呢?

动态多态: 显然这和静态多态是一组反义词,它是在程序运行时根据基类的引用(指针)指向的对象来确定自己具体该调用哪一个类的虚函数。

我在西安临潼上学,我就以这边的公交车举个栗子啊:

class TakeBus

{

public:

    void TakeBusToSubway()

    {

        cout << "go to Subway--->please take bus of 318" << endl;

    }

    void TakeBusToStation()

    {

        cout << "go to Station--->pelase Take Bus of 306 or 915" << endl;

    }

};

//知道了去哪要做什么车可不行,我们还得知道有没有这个车

class Bus

{

public:

    virtual void TakeBusToSomewhere(TakeBus& tb) = 0;  //???为什么要等于0

};

class Subway:public Bus

{

public:

    virtual void TakeBusToSomewhere(TakeBus& tb)

    {

        tb.TakeBusToSubway();

    }

};

class Station :public Bus

{

public:

    virtual void TakeBusToSomewhere(TakeBus& tb)

    {

        tb.TakeBusToStation();

    }

};

int main()

{

    TakeBus tb;

    Bus* b = NULL;

    //假设有十辆公交车,如果是奇数就是去地铁口的,反之就是去火车站的

    for (int i = 1; i <= 10; ++i)

    {

        if ((rand() % i) & 1)

            b = new Subway;

        else

            b = new Station;

    }

    b->TakeBusToSomewhere(tb);

    delete b;

    return 0;

}

这就是一个简单的动态多态的例子,它是在程序运行时根据条件去选择调用哪一个函数。

而且,从上面的例子我们还发现了我在每一个函数前都加了virtual这个虚拟关键字,想想为什么?如果不加会不会构成多态呢?

C/C++学习交流QQ群728483370,大家一起加油!

干想不如上机实践:

class Base

{

public:

    virtual void Funtest1(int i)

    {

        cout << "Base::Funtest1()" << endl;

    }

    void Funtest2(int i)

    {

        cout << "Base::Funtest2()" << endl;

    }

};

class Drived :public Base

{

    virtual void Funtest1(int i)

    {

        cout << "Drived::Fubtest1()" << endl;

    }

    virtual void Funtest2(int i)

    {

        cout << "Drived::Fubtest2()" << endl;

    }

    void Funtest2(int i)

    {

        cout << "Drived::Fubtest2()" << endl;

    }

};

void TestVirtual(Base& b)

{

    b.Funtest1(1);

    b.Funtest2(2);

}

int main()

{

    Base b;

    Drived d;

    TestVirtual(b);

    TestVirtual(d);

    return 0;

}

在调用FuncTest2的时候我们看出来他并没有给我们调用派生类的函数,因此我们可以对动态多态的实现做个总结。

动态多态的条件:

●基类中必须包含虚函数,并且派生类中一定要对基类中的虚函数进行重写。

●通过基类对象的指针或者引用调用虚函数。

重写 :

(a)基类中将被重写的函数必须为虚函数(上面的检测用例已经证实过了)

(b)基类和派生类中虚函数的原型必须保持一致(返回值类型,函数名称以及参数列表),协变和析构函数(基类和派生类的析构函数是不一样的)除外

(c)访问限定符可以不同

那么问题又来了,什么是协变?

协变:基类(或者派生类)的虚函数返回基类(派生类)的指针(引用)

总结一道面试题:那些函数不能定义为虚函数?

经检验下面的几个函数都不能定义为虚函数:

1)友元函数,它不是类的成员函数

2)全局函数

3)静态成员函数,它没有this指针

3)构造函数,拷贝构造函数,以及赋值运算符重载(可以但是一般不建议作为虚函数)

抽象类:

在前面公交车的例子上我提了一个问题:

class Bus

{

public:

    virtual void TakeBusToSomewhere(TakeBus& tb) = 0;  //???为什么要等于0

};

在成员函数(必须为虚函数)的形参列表后面写上=0,则成员函数为纯虚函数。包含纯虚函数的类叫做抽象类(也叫接口类),抽象类不能实例化出对象。纯虚函数在派生类中重新定义以后,派生类才能实例化出对象。纯虚函数是一定要被继承的,否则它存在没有任何意义。

多态调用原理

class Base

{

public:

    virtual void Funtest1(int i)

    {

        cout << "Base::Funtest1()" << endl;

    }

    virtual void Funtest2(int i)

    {

        cout << "Base::Funtest2()" << endl;

    }

    int _data;

};

int main()

{

    cout << sizeof(Base) << endl;

    Base b;

    b._data = 10;

    return 0;

}

8?不知道大家有没有问题,反正我是有疑惑了。以前在对象模型(https://blog.csdn.net/qq_39412582/article/details/80808754)时我提到过怎么来求一个类的大小。按照那个方法,这里应该是4才对啊,为什么会是8呢?

通过观察。我们发现这个例子里面和以前不一样,类成员函数变成了虚函数,这是不是引起类大小变化的原因呢?

我们假设就是这样,然后看看内存里是怎么存储的呢?

可以看到它在内存里多了四个字节,那这四个字节的内容到底是什么呢?

是不是有点看不懂,我们假设它是一个地址去看地址里存的东西的时候发现它存的是两个地址。

我假设它是虚函数的地址,我们来验证一下:

typedef void (__stdcall *PVFT)();  //函数指针

int main()

{

    cout << sizeof(Base) << endl;

    Base b;

    b._data = 10;

    PVFT* pVFT = (PVFT*)(*((int*)&b));

    while (*pVFT)

    {

        (*pVFT)();

        pVFT+=1;

    }

    return 0;

}

结果好像和我们的猜想一样,是一件开心的事。然后我给一张图总结一下:

在反汇编中我们还可以看到,如果含有虚函数的类中没有定义构造函数,编译器会自动合成一个构造函数

对于派生类的东西我给个链接仔细看,人家总结的超级赞,我偷个懒就不写了,老铁们包容下啊。

派生类虚表:

1.先将基类的虚表中的内容拷贝一份

2.如果派生类对基类中的虚函数进行重写,使用派生类的虚函数替换相同偏移量位置的基类虚函数

3.如果派生类中新增加自己的虚函数,按照其在派生类中的声明次序,放在上述虚函数之后

4.C/C++学习交流QQ群728483370,大家一起加油!

多态缺陷

●降低了程序运行效率(多态需要去找虚表的地址)

●空间浪费

相关文章

  • C++ 之 多态(非常非常重要,重点在后面)

    编译环境:WIN10 VS2017 这篇文章有点长,但都是满满的干货,一定要看到最后,那才是重点。 什么是多态? ...

  • 深入研究C++多态(虚函数和虚继承)

    面向对象的三大特性是封装、继承和多态。多态是非常重要的一个特性,C++多态基于虚函数和虚继承实现,本文将完整挖掘C...

  • 深刻剖析之c++博客文章

    三大特性 封装、继承、多态 多态 C++ 虚函数表解析C++多态的实现原理 介绍了类的多态(虚函数和动态/迟绑定)...

  • 26号c#总结

    26号 今天开始讲多态,在c和c++中都有涉及。继承是子类使用父类的方法,而多态则是父类使用子类的方法。重点内容是...

  • C++学习笔记(3)

    摘要:本篇对C++的三个重点「封装,继承,多态」的学习总结与使用。 很遗憾,在坚持了一段时间c++的学习后,有点想...

  • C++入门06 --多态,虚函数,虚函数表,纯虚函数,抽象类

    多态 默认情况下,编译器只会根据指针类型调用对应的函数,不存在多态; 多态是面向对象的非常重要的一个特性;同一操作...

  • 多态、虚函数、虚函数表

    多态 默认情况下,编译器只会根据指针类型调用对应的函数,不存在多态 多态是面向对象非常重要的一个特性同一操作作用于...

  • 多态的C++实现

    多态的C++实现 1 多态的原理 什么是多态?多态是面向对象的特性之一,只用父类指针指向子类的对象。 1.1 多态...

  • C++ 的多态(Polymorphism), virtual f

    多态 c++支持两种多态,编译时多态和运行时多态但其实编译时多态根本不是真正的多态,编译时多态就是函数的重载,ov...

  • 【6.23】java的继承与多态

    java 继承 多态 继承与多态是面向对象的语言的两个重要的特点,深入的认识对运用好这门编程语言非常重要。 今天的...

网友评论

      本文标题:C++ 之 多态(非常非常重要,重点在后面)

      本文链接:https://www.haomeiwen.com/subject/twijdctx.html