C++ 重载、隐藏、覆盖的区别小结

 更新时间:2026年03月20日 09:43:32   作者:越甲八千  
本文通过C++代码示例深入解析了重载(Overload)、覆盖(Override)和隐藏(Hide)三个核心概念的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

结合这C++ 代码,深入理解重载(Overload)隐藏(Hide)覆盖(Override,也叫重写) 这三个核心概念,用这段代码作为核心案例,由浅入深地拆解它们的区别和本质。

一、先明确三个概念的核心定义

这三个概念都是 C++ 中函数名复用的场景,但适用范围、规则和目的完全不同,先看一张对比表建立整体认知:

概念中文名称适用范围核心特征本质目的
Overload重载同一个作用域内函数名相同,参数列表(个数/类型/顺序)不同,返回值不影响方便同一功能的不同参数调用
Override覆盖/重写子类与父类之间子类重写父类的虚函数,函数签名(名+参数+const)完全一致实现多态(运行时动态绑定)
Hide隐藏/遮蔽子类与父类之间子类函数名覆盖父类同名函数(无论参数/虚函数与否),编译期静态屏蔽作用域优先原则导致的“遮蔽”

覆盖是特殊的隐藏。

二、结合代码逐个拆解

逐个分析这三个概念,并用扩展代码演示效果。

1. 重载(Overload)—— 同一作用域内的“同名不同参”

核心规则:

  • 必须在同一个类/同一个作用域中;
  • 函数名完全相同;
  • 参数列表必须不同(个数、类型、顺序,三者至少一个不同);
  • 返回值、是否是虚函数不影响重载(但仅返回值不同不算重载)。

代码中的重载案例:

Base 类中,两个 show 函数就是典型的重载

class Base  
{
public:
    Base(int a):ma(a) {}
    // 重载1:无参 show
    virtual void show() {cout<<"Base::show()"<<endl;}
    // 重载2:带int参数的 show —— 与上面构成重载
    virtual void show(int i) {cout<<"Base::show(int)"<<endl;}
private:
    int ma;
};
  • 它们在 Base 这个同一作用域内;
  • 函数名都是 show
  • 参数列表不同(一个无参,一个int参数);
  • 都是虚函数(但即使不是虚函数,依然是重载)。

重载的调用示例:

int main() {
    Base b(10);
    b.show();    // 调用 Base::show() —— 匹配无参版本
    b.show(5);   // 调用 Base::show(int) —— 匹配int参数版本
    return 0;
}

编译期编译器会根据实参类型自动匹配对应的重载函数,这是静态绑定(编译期确定调用哪个)。

2. 覆盖/重写(Override)—— 子类对父类虚函数的“精准替换”

核心规则:

  • 必须在子类与父类之间;
  • 父类函数必须是虚函数(virtual)
  • 子类函数的函数签名完全一致(函数名 + 参数列表 + const/volatile 属性完全相同);
  • 子类函数可以加 override 关键字显式声明(C++11 后推荐,编译器会检查是否真的覆盖)。

代码中的覆盖案例:

Derive 类中的 show() 覆盖了 Base 类中的无参 show()

class Derive : public Base
{
public:
    Derive(int a, int b):Base(a), mb(b) {}
    // 覆盖:与 Base::show() 签名完全一致,且父类是虚函数
    void show() override {cout<<"Derive::show()"<<endl;} // 加override更规范
private:
    int mb;
};
  • 父类 Base::show() 是虚函数;
  • 子类 Derive::show() 函数名、参数列表(无参)完全一致;
  • 这就是“覆盖”,目的是实现多态

覆盖的多态调用示例:

int main() {
    Base* ptr = new Derive(1, 2); // 父类指针指向子类对象
    ptr->show();  // 调用 Derive::show() —— 运行时动态绑定(多态)
    delete ptr;
    return 0;
}

输出:Derive::show()
这里因为覆盖了虚函数,运行时会根据对象的实际类型(Derive)调用对应的函数,而不是指针类型(Base),这是动态绑定

3. 隐藏/遮蔽(Hide)—— 子类对父类同名函数的“编译期屏蔽”

核心规则:

  • 必须在子类与父类之间;
  • 子类定义了与父类同名的函数(无论参数是否相同、是否是虚函数);
  • 编译器在编译子类作用域时,会屏蔽父类所有同名函数,即使父类是重载/虚函数;
  • 这是编译期静态行为,与多态无关。

代码中的隐藏案例(重点!):

你的代码中,Derive 类定义了 show(),会隐藏父类 Base 中所有名为 show 的函数(包括 show()show(int)):

int main() {
    Derive d(1, 2);
    d.show();    // 正常:调用 Derive::show()
    // d.show(5); // 编译错误!父类的 show(int) 被隐藏了
    return 0;
}

为什么会报错?
因为子类 Derive 中有 show() 函数,编译器在解析 d.show(5) 时,会优先在 Derive 作用域找 show,发现只有无参的 show(),没有 show(int),且不会去父类找(被隐藏了),因此编译失败。

隐藏的其他场景(补充理解):

即使子类函数参数与父类不同,依然会隐藏:

class Derive : public Base
{
public:
    Derive(int a, int b):Base(a), mb(b) {}
    // 子类定义 show(double) —— 与父类 show(int) 不同参,但依然隐藏父类所有 show
    void show(double d) {cout<<"Derive::show(double)"<<endl;}
private:
    int mb;
};

int main() {
    Derive d(1,2);
    // d.show(5); // 编译错误!父类 show(int) 被隐藏,子类只有 show(double)
    d.show(5.0); // 正常:调用 Derive::show(double)
    return 0;
}

如何解除隐藏?

如果想在子类中调用父类被隐藏的函数,有两种方式:

  1. using 声明,把父类函数引入子类作用域:
class Derive : public Base
{
public:
    using Base::show; // 解除父类所有 show 函数的隐藏
    Derive(int a, int b):Base(a), mb(b) {}
    void show() override {cout<<"Derive::show()"<<endl;}
private:
    int mb;
};

// 此时调用正常
int main() {
    Derive d(1,2);
    d.show();    // 调用 Derive::show()
    d.show(5);   // 调用 Base::show(int) —— 隐藏被解除
    return 0;
}
  1. 显式指定父类作用域调用:
int main() {
    Derive d(1,2);
    d.Base::show(5); // 直接调用父类的 show(int),不受隐藏影响
    return 0;
}

三、容易混淆的点对比

1. 覆盖 vs 重载

维度覆盖(Override)重载(Overload)
作用域跨类(父类→子类)同作用域(同一类)
函数签名必须完全一致必须不同(参数列表)
虚函数要求父类必须是虚函数无要求(可虚可非虚)
绑定方式运行时动态绑定编译期静态绑定

2. 覆盖 vs 隐藏

维度覆盖(Override)隐藏(Hide)
前提父类是虚函数 + 签名一致只要子类有同名函数(无论参数/虚函数)
本质多态的实现方式作用域优先导致的屏蔽
绑定方式运行时动态绑定编译期静态屏蔽
父类函数可用性父类虚函数被“替换”但可通过作用域调用父类同名函数被屏蔽,需显式调用

四、完整演示代码

把上述所有场景整合,可以直接运行测试:

#include <iostream>
using namespace std;

class Base  
{
public:
    Base(int a):ma(a) {}
    // 重载1:无参虚函数
    virtual void show() {cout<<"Base::show()"<<endl;}
    // 重载2:带int参数的虚函数(与上面构成重载)
    virtual void show(int i) {cout<<"Base::show(int) "<<i<<endl;}
private:
    int ma;
};

class Derive : public Base
{
public:
    using Base::show; // 解除父类show的隐藏(注释掉这行看隐藏效果)
    Derive(int a, int b):Base(a), mb(b) {}
    // 覆盖:重写父类无参show
    void show() override {cout<<"Derive::show()"<<endl;}
    // 新增:子类独有的show(double)(会隐藏父类show,除非加using)
    void show(double d) {cout<<"Derive::show(double) "<<d<<endl;}
private:
    int mb;
};

int main() {
    // 1. 测试Base的重载
    Base b(10);
    b.show();     // Base::show()
    b.show(5);    // Base::show(int) 5

    // 2. 测试多态(覆盖)
    Base* ptr = new Derive(1, 2);
    ptr->show();  // Derive::show() —— 动态绑定
    ptr->show(6); // Base::show(int) 6 —— 子类未覆盖该版本,调用父类
    delete ptr;

    // 3. 测试子类的隐藏与解除
    Derive d(1, 2);
    d.show();     // Derive::show()
    d.show(7);    // Base::show(int) 7(加using后生效,否则编译错误)
    d.show(3.14); // Derive::show(double) 3.14
    d.Base::show(); // 显式调用父类show()

    return 0;
}

总结

  1. 重载是同一作用域内“同名不同参”,编译期静态匹配,目的是简化调用;
  2. 覆盖是子类重写父类虚函数(签名一致),运行时动态绑定,是多态的核心;
  3. 隐藏是子类同名函数屏蔽父类所有同名函数(编译期行为),需用 using 或作用域显式调用父类函数。

这三个概念的核心区别在于作用域绑定时机:重载是“同作用域+编译期”,覆盖是“跨类+运行期”,隐藏是“跨类+编译期屏蔽”。

补充

#include <iostream>
using namespace std;

class Base  // Base base(10);
{
public:
    Base(int a):ma(a) {}
    // 虚函数
    virtual void show() {cout<<"Base::show()"<<endl;}
    virtual void show(int i) {cout<<"Base::show(int)"<<endl;}
private:
    int ma;
};

class Derive : public Base
{
public:
    Derive(int a, int b):Base(a), mb(b) {}
    void show() {cout<<"Derive::show"<<endl;}
private:
    int mb;
};

补充说明

  1. 虚函数与重写

    • Base 类中定义了两个虚函数 show()show(int i)
    • Derive 类重写了无参版本 show(),但没有重写带参版本 show(int i)
    • 由于 C++ 的名称隐藏规则Derive 类中定义的 show() 会隐藏掉基类中所有名为 show 的函数(包括 show(int i))。
  2. 典型调用行为

    int main() {
        Derive d(1, 2);
        d.show();        // 正确,调用 Derive::show()
        // d.show(10);   // 编译错误!基类的 show(int) 被隐藏了
        return 0;
    }
    

    如果想在 Derive 中也能调用 Base::show(int),需要在 Derive 中添加 using Base::show; 来解除名称隐藏。

到此这篇关于C++ 重载、隐藏、覆盖的区别小结的文章就介绍到这了,更多相关C++ 重载、隐藏、覆盖内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++小知识:用合适的工具来分析你的代码

    C++小知识:用合适的工具来分析你的代码

    今天小编就为大家分享一篇关于C++小知识:用合适的工具来分析你的代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • 在vs2010中,输出当前文件路径与源文件当前行号的解决方法

    在vs2010中,输出当前文件路径与源文件当前行号的解决方法

    本篇文章是对在vs2010中,输出当前文件路径与源文件当前行号的解决方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++ 将一个文件读入数组再读出数组的方法

    C++ 将一个文件读入数组再读出数组的方法

    今天小编就为大家分享一篇C++ 将一个文件读入数组再读出数组的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • 一道面试题教你轻松玩转C++指针

    一道面试题教你轻松玩转C++指针

    下面小编就为大家带来一篇深入理解c++指针的指针和指针的引用。小编觉得挺不错的,现在就分享给大家,也给大家做个参考,一起跟随小编过来看看吧
    2021-09-09
  • C++填坑的重写,重载和隐藏的详解

    C++填坑的重写,重载和隐藏的详解

    这篇文章主要介绍了C++中重载、重写(覆盖)和隐藏的区别,是C++面向对象程序设计非常重要的概念,需要的朋友可以参考下,希望能够给你带来帮助
    2021-09-09
  • 成员初始化列表与构造函数体中的区别详细解析

    成员初始化列表与构造函数体中的区别详细解析

    无论是在构造函数初始化列表中初始化成员,还是在构造函数体中对它们赋值,最终结果是相同的。不同之处在于,使用构造函数初始化列表的版本初始化数据成员,没有定义初始化列表的构造函数版本在构造函数体中对数据成员赋值
    2013-09-09
  • 彻底掌握C语言strcpy函数的用法

    彻底掌握C语言strcpy函数的用法

    C语言中的strcpy函数,是一种C语言的标准库函数,它用于对字符串进行复制。本章带你了解它的使用并模拟实现它
    2022-05-05
  • C++中priority_queue模拟实现的代码示例

    C++中priority_queue模拟实现的代码示例

    在c++语言中数据结构中的堆结构可以通过STL库中的priority_queue 优先队列来实现,这样做极大地简化了我们的工作量,这篇文章主要给大家介绍了关于C++中priority_queue模拟实现的相关资料,需要的朋友可以参考下
    2021-08-08
  • VSstudio中scanf返回值被忽略的原因及解决方法(推荐)

    VSstudio中scanf返回值被忽略的原因及解决方法(推荐)

    这篇文章主要介绍了VSstudio中scanf返回值被忽略的原因及其解决方法,scanf返回值被忽略,接下来我就告诉大家该如何解决这个问题,需要的朋友可以参考下
    2022-09-09
  • C++哈希应用的位图和布隆过滤器

    C++哈希应用的位图和布隆过滤器

    这篇文章主要介绍了C++哈希应用的位图和布隆过滤器的相关资料,文章内容多以列举试题的方式讲解,感兴趣的朋友可以参考下面文章内容
    2021-09-09

最新评论