c++中虚函数、纯虚函数以及虚函数的实现原理

c++中虚函数、纯虚函数以及虚函数的实现原理

什么是虚函数和纯虚函数

虚函数(Virtual Functions)和纯虚函数(Pure Virtual Functions)是 C++ 中用于实现多态性的重要概念。

虚函数(Virtual Functions)

虚函数是在基类中声明为虚函数的成员函数。它允许在派生类中进行重写(覆盖),并且在运行时根据对象的实际类型调用适当的函数。虚函数通过使用 virtual 关键字进行声明,派生类可以选择性地重写基类中的虚函数。

示例:
#include <iostream>

class Base {
public:
    virtual void show() {
        std::cout << "Base::show()" << std::endl;
    }
};

class Derived : public Base {
public:
    void show() override {
        std::cout << "Derived::show()" << std::endl;
    }
};

int main() {
    Base* basePtr = new Derived();
    basePtr->show(); // 输出 Derived::show()
    delete basePtr;
    return 0;
}

在这个例子中,Base 类中的 show() 函数被声明为虚函数,在 Derived 类中对其进行了重写。当使用基类指针指向派生类对象时,调用 show() 函数时会根据对象的实际类型来调用适当的函数。

纯虚函数(Pure Virtual Functions)

纯虚函数是在基类中声明为纯虚函数的虚函数,它没有具体的实现,而是用 = 0 来指示编译器该函数没有实现。含有纯虚函数的类被称为抽象类,不能直接实例化,只能作为基类来派生其他类。

示例:
#include <iostream>

class Shape {
public:
    // 纯虚函数
    virtual void draw() = 0;
};

class Circle : public Shape {
public:
    // 实现了纯虚函数
    void draw() override {
        std::cout << "Drawing a circle." << std::endl;
    }
};

int main() {
    // Shape shape; // 错误!抽象类不能被实例化
    Circle circle;
    circle.draw(); // 输出 Drawing a circle.
    return 0;
}

在这个例子中,Shape 类中的 draw() 函数被声明为纯虚函数,因此 Shape 类成为了一个抽象类。Circle 类继承自 Shape 类,并且实现了 draw() 函数,因此 Circle 类可以被实例化。

虚函数的实现原理

在 C++ 中,虚函数的实现原理涉及到两个关键概念:虚函数表(vtable)和虚函数指针(vptr)。

1. 虚函数表(vtable)

虚函数表是一张存储了类中虚函数地址的表格,每个包含虚函数的类都会有一个对应的虚函数表。虚函数表中的每个条目存储了一个虚函数的地址,通常是类中声明的虚函数的地址。

当类中至少有一个虚函数时,编译器会为该类生成一个虚函数表。每个类只有一个虚函数表,存储在内存的一个固定位置。子类的虚函数表会包含其继承的父类的虚函数表,以及自己新增的虚函数。

2. 虚函数指针(vptr)

虚函数指针是一个指向虚函数表的指针,它存储在类的对象中。每个类的对象都有一个对应的虚函数指针,指向该类的虚函数表。

当调用一个虚函数时,编译器会使用对象中的虚函数指针找到对应的虚函数表,然后在虚函数表中查找相应的虚函数地址,并调用该函数。

示例

#include <iostream>

class Base {
public:
    virtual void show() {
        std::cout << "Base::show()" << std::endl;
    }
};

class Derived : public Base {
public:
    void show() override {
        std::cout << "Derived::show()" << std::endl;
    }
};

int main() {
    Base* basePtr = new Derived();
    basePtr->show(); // Derived::show()
    delete basePtr;
    return 0;
}

在这个例子中,Base 类有一个虚函数 show()Derived 类继承自 Base 类并重写了 show() 函数。当创建 Derived 类的对象并将其赋值给 Base 类的指针时,实际上是将 basePtr 指向了 Derived 类的对象。当调用 basePtr->show() 时,虚函数机制会使用 basePtr 中存储的虚函数指针找到 Derived 类的虚函数表,然后调用 Derived 类中的 show() 函数。

虚函数机制通过虚函数表和虚函数指针实现了运行时多态性,允许在运行时根据对象的实际类型来调用适当的函数。

  • 10
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值