C++ 虚函数(多态,多重继承,菱形继承)

多态的实现依赖于虚函数。类存在虚函数,则编译器会为每个类创建虚函数表。

虚函数的调用过程:创建基类赋值给基类指针 --> 虚函数表指针 --> 虚函数表(数组) 里面进行查询并调用实际的函数。

基本实现

cpp 复制代码
#include <iostream>

// 基类
class Base {
public:
	// 虚函数1
    virtual void func1() { std::cout << "Base::func1()" << std::endl; }
    // 虚函数2
    virtual void func2() { std::cout << "Base::func2()" << std::endl; }
};

// 派生类 继承
class Derived : public Base {
public:
    // 重写
    void func1() override { std::cout << "Derived::func1()" << std::endl; }
};

int main() {
    Base* ptr = new Derived();


    // 通过基类指针调用虚函数,运行时动态绑定
    ptr->func1();  // 输出: Derived::func1()
    ptr->func2();  // 输出: Base::func2()
    
    delete ptr;
    return 0;
}

多重继承

派生类继承了多个基类,那就存在多个虚函数指针。

cpp 复制代码
#include <iostream>

// 基类A
class BaseA {
public:
    virtual void funcA() { std::cout << "BaseA::funcA()" << std::endl; }
    virtual void common() { std::cout << "BaseA::common()" << std::endl; }
};

// 基类B
class BaseB {
public:
    virtual void funcB() { std::cout << "BaseB::funcB()" << std::endl; }
    virtual void common() { std::cout << "BaseB::common()" << std::endl; }
};

// 派生类:多重继承自BaseA和BaseB
class Derived : public BaseA, public BaseB {
public:
    void funcA() override { std::cout << "Derived::funcA()" << std::endl; }
    void funcB() override { std::cout << "Derived::funcB()" << std::endl; }
    void common() override { std::cout << "Derived::common()" << std::endl; }
};

int main() {
    Derived d;
    
    // 通过不同基类指针调用虚函数
    BaseA* ptrA = &d;
    BaseB* ptrB = &d;
    
    ptrA->funcA();   // 输出: Derived::funcA()
    ptrA->common();  // 输出: Derived::common()
    
    ptrB->funcB();   // 输出: Derived::funcB()
    ptrB->common();  // 输出: Derived::common()
    
    // 直接通过派生类对象调用
    d.funcA();       // 输出: Derived::funcA()
    d.funcB();       // 输出: Derived::funcB()
    d.common();      // 输出: Derived::common()
    
    return 0;
}

菱形继承

继承关系:

A

/

B C

\ /

D

虚基表

当前例子:D里面会有B和C的虚函数表,当调用公共的虚函数时,会进行进行跳转。

cpp 复制代码
#include <iostream>

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

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

class C : virtual public A {
public:
    void foo() override { std::cout << "C::foo()" << std::endl; }
};

class D : public B, public C {
public:
    void foo() override { std::cout << "D::foo()" << std::endl; }
};

int main() {
    D d;
    A* ptr = &d;
    ptr->foo();  // 输出: D::foo()
    
    return 0;
}
相关推荐
源代码•宸15 小时前
Golang面试题库(Interface、GMP)
开发语言·经验分享·后端·面试·golang·gmp·调度过程
西京刀客15 小时前
Go 语言中的 toolchain 指令-toolchain go1.23.6的作用和目的
开发语言·后端·golang·toolchain
weisian15115 小时前
JVM--2-打破刻板印象:在Java中创建一个对象,一定是分配到堆内存吗?
java·开发语言·jvm·tlab·逃逸分析·标量替换
心语星光15 小时前
用python语言的pyautogui库实现伪批量将xdf文件打印为pdf文件
开发语言·python·pdf·自动化
cyforkk15 小时前
10、Java 基础硬核复习:多线程(并发核心)的核心逻辑与面试考点
java·开发语言·面试
2301_8223827615 小时前
嵌入式C++实时内核
开发语言·c++·算法
Max_uuc15 小时前
【C++ 硬核】拒绝单位混淆:利用 Phantom Types (幻影类型) 实现零开销的物理量安全计算
开发语言·c++
Remember_99315 小时前
Java 工厂方法模式:解耦对象创建的优雅方案
java·开发语言·python·算法·工厂方法模式
2301_7903009615 小时前
C++与物联网开发
开发语言·c++·算法
A懿轩A15 小时前
【Java 基础编程】Java 运算符完全指南:算术/关系/逻辑/位运算与优先级,避免常见坑
java·开发语言