C++ 多态中虚函数实现原理分析

本文详细解释了C++中多态特性的实现原理,通过示例代码展示了如何使用虚函数实现多态,并深入探讨了虚函数表的工作机制。

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

C++ 多态

多态在C++中表现为,通过基类指针指向其派生类对象,用该指针调用基类中声明的虚函数,编译器可以自动根据该对象类别来确定实际调用的函数。可以通过以下例子说明:

//demo1
#include <iostream>
using namespace std;

class A
{
public:
    A(){}
    void say(){cout<<"A\n";}
};

class B:public A
{
public:
    B(){}
    void say(){cout<<"B\n";}
};

int main()
{
    A *ptr = new B;
    ptr->say();
    delete ptr;
    return 0;
}

输出结果 果断是

>A

。因为没加virtual关键字声明为虚函数

如果在A(基类)中将say函数声明为

virtual void say(){cout<<"A\n";} //多态实现中最为关键的一个地方

此处将会输出B.


C++ 中多态特性的实现就是通过虚函数来实现的,虚函数的作用就是像上面所表示的那样,那虚函数的特性又是如何实现的呢。

◆ 上面代码中的A类,加了virtual与不加virtual关键字 实际上大小是不一样的。加了virtual的类中编译器会为其加上一个看不见的成员——一个指向虚函数表的指针,该成员在该类对象的基地址处(最低地址)。在64位系统中通常占8字节。

◆ 派生类对象在内存上是如何构成的,下面是一个 B类型对象的内容存储空间 B类型继承自A类型,A类型中say函数为虚函数

        —————————— 低地址 B类型对象基地址

        |      A--base class      |

        ——————————              假设这个对象叫 b

        |      B--child class      |

        —————————— 高地址


        就是这样的结构。

◆ 一个类的派生类如果重写了其父类中声明的虚函数,那么该派生类所声明的对象中的父类的区域中的虚函数表就会被修改为指向派生类中重写的新的函数。


由以上三点可知

◆ b(B类型对象) 的父类(A类型)区域的第一个成员就是 虚函数表的指针

◆ b的B类型区域的第一个成员不是虚函数表指针,因为B类型中并没有自己独有的虚函数声明

◆ 用A类型的指针 (A *)指向B类型声明的对象等于截去了下面B类型的部分。那此时如果没有虚函数表,没有虚函数指针,通过(A*)来进行函数调用,调用的肯定就是A类型的函数。

◆ 通过第三点可以得出结论,b对象的A类型区域是含有虚函数表的,那么在调用say函数的时候,会通过虚函数表找到B类型区域中的say函数,从而实现了C++的多态特性。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值