当前位置: 首页 > news >正文

C++ 重载(Overload)、重写(Override)、隐藏(Hiding) 的区别

C++ 重载(Overload)、重写(Override)、隐藏(Hiding) 的区别

这三个概念是 C++ 面向对象的核心知识点,也是面试必问内容。下面我们从定义、发生条件、代码示例、底层原理全方位解析它们的区别。

一、核心区别对比表(速记版)

特性重载(Overload)重写(Override)隐藏(Hiding)
作用域同一作用域父子类继承关系父子类继承关系
函数名相同相同相同
参数列表必须不同必须相同可以相同或不同
virtual不需要必须基类有virtual不需要
返回值可不同必须相同(协变除外)可不同
多态性编译时决定运行时多态编译时决定

二、重载(Overload) - 同一作用域的函数变体

定义

同一作用域内,函数名相同但参数列表不同(类型、顺序、数量)

关键代码示例

class Calculator {
public:// 重载示例int add(int a, int b) { return a + b; }double add(double a, double b) { return a + b; }int add(int a, int b, int c) { return a + b + c; }
};

核心特点

  1. 必须在同一作用域(如同一个类中)
  2. 返回值类型可不同(但仅返回值不同不构成重载!)
  3. 调用时编译器根据参数类型决定调用哪个版本
  4. 典型应用:构造函数重载、运算符重载

三、重写(Override) - 多态性的核心实现

定义

继承体系中,派生类重新定义基类virtual 函数,实现运行时多态

关键代码示例

class Animal {
public:virtual void speak() { cout << "Animal sound" << endl; } // 虚函数
};class Dog : public Animal {
public:void speak() override { cout << "Wang Wang!" << endl; } // 重写
};Animal* animal = new Dog();
animal->speak(); // 输出 "Wang Wang!"(多态调用)

核心特点

  1. 必须通过基类指针/引用调用才能体现多态
  2. 函数签名必须完全相同(C++11可用override关键字检查)
  3. 特殊情况:协变返回类型(允许返回派生类指针/引用)
    class Base {
    public:virtual Base* clone() { return new Base(); }
    };
    class Derived : public Base {
    public:Derived* clone() override { return new Derived(); } // 合法协变
    };
    

四、隐藏(Hiding) - 最容易踩坑的隐藏规则

定义

当派生类定义了与基类同名函数(无论参数是否相同),隐藏基类的同名函数。

关键代码示例

class Base {
public:void func() { cout << "Base::func()" << endl; }void func(int) { cout << "Base::func(int)" << endl; }
};class Derived : public Base {
public:void func() { cout << "Derived::func()" << endl; } // 隐藏基类所有func版本
};Derived d;
d.func();    // 正确,调用Derived::func()
// d.func(1); // 编译错误!基类func(int)被隐藏

核心特点

  1. 隐藏所有基类同名函数(包括重载版本)
  2. 可通过using声明恢复可见性:
    class Derived : public Base {
    public:using Base::func; // 恢复基类func的所有重载void func() { cout << "Derived::func()" << endl; }
    };
    
  3. 与重写的区别:不需要virtual不构成多态

五、对比案例分析

案例1:重载 vs 重写

class A {
public:virtual void foo(int) { cout << "A::foo(int)" << endl; }void bar(int) { cout << "A::bar(int)" << endl; }
};class B : public A {
public:void foo(int) override { cout << "B::foo(int)" << endl; } // 重写void bar(double) { cout << "B::bar(double)" << endl; }    // 隐藏(非重载!)
};B b;
A* pa = &b;
pa->foo(1);   // B::foo(int) (多态)
pa->bar(1);   // A::bar(int) (无多态)
b.bar(1.0);   // B::bar(double)
// b.bar(1);   // 编译警告,隐式转换,仍然调用B::bar(double)

案例2:隐藏的陷阱

class Base {
public:void print(int x) { cout << "Base: " << x << endl; }
};class Derived : public Base {
public:void print(string s) { cout << "Derived: " << s << endl; } // 隐藏基类print(int)
};Derived d;
d.print("hello"); // Derived: hello
// d.print(42);   // 编译错误!基类print(int)被隐藏
d.Base::print(42); // 正确,显式指定作用域

六、面试高频问题

Q1:如何强制检查是否正确重写?

✅ C++11 使用 override 关键字:

class Child : public Parent {void foo() override; // 如果Parent没有virtual foo(),编译报错
};

Q2:重载能否跨作用域?

❌ 不能!派生类定义同名函数会隐藏基类重载版本(除非使用using

Q3:虚函数能否重载?

✅ 可以!但重写时必须签名完全匹配(协变返回除外)

Q4:如何区分重写和隐藏?

✅ 看两点:

  1. 基类是否有virtual关键字
  2. 是否通过基类指针/引用调用表现出多态
http://www.xdnf.cn/news/742321.html

相关文章:

  • LiquiGen流体导入UE
  • STM32 HAL库函数学习 CRC篇
  • Linux系统编程之共享内存
  • 在QT中,利用charts库绘制FFT图形
  • MAC软件游戏打开提示已损坏
  • MATLAB实战:机器学习分类回归示例
  • 【MFC】如何设置让exe的控制台不会跟着exe退出而退出
  • C++中指针常量和常量指针的区别
  • 【设计模式-4.6】行为型——状态模式
  • [蓝桥杯]拉马车
  • L56.【LeetCode题解】 电话号码的字母组合
  • 触发器与存储过程详解
  • Mybatis-Plus简单介绍
  • 鸿蒙HarmonyOS (React Native)的实战教程
  • Java后端技术栈问题排查实战:Spring Boot启动慢、Redis缓存击穿与Kafka消费堆积
  • 【Java学习笔记】内部类(重点)
  • 数据结构:时间复杂度(Time Complexity)和空间复杂度(Space Complexity)
  • Typescript学习教程,从入门到精通,TypeScript 配置管理与编译器详解(19)
  • Rust 配置解析`serde` + `toml`
  • 华为OD机试真题——找出两个整数数组中同时出现的整数(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
  • 【HTML】基础学习【数据分析全栈攻略:爬虫+处理+可视化+报告】
  • MySQL事务与锁机制详解:确保数据一致性的关键【MySQL系列】
  • 005 flutter基础,初始文件讲解(4)
  • leetcode付费题 353. 贪吃蛇游戏解题思路
  • 实现MPC钱包
  • [蓝桥杯]阶乘求值【省模拟赛】
  • Thinkphp6实现websocket
  • Spring Boot养老院管理系统源码分享
  • MacOS安装Docker Desktop并汉化
  • 2.5 TypeScript 中的 instanceof 运算符