函数重载发生在同一作用域内,参数列表不同,编译时确定;重写需基类virtual函数,派生类同名同参函数实现多态,运行时确定;隐藏是派生类同名函数屏蔽基类所有同名函数,无论参数是否相同,静态类型决定调用。

在C++中,函数重载、重写(覆盖)和隐藏是三个容易混淆的概念,它们分别发生在不同的作用域和继承关系中,影响着多态行为和函数调用的解析方式。理解这三者的区别对掌握C++面向对象编程和多态机制至关重要。
函数重载(Overloading)
函数重载指的是在同一个作用域内,多个函数具有相同的名称但参数列表不同(参数个数、类型或顺序不同),编译器根据调用时传入的实参来选择匹配的函数版本。
重载不依赖于继承,也不涉及virtual关键字。它是静态联编(编译时决定)的典型例子。
示例:class Math {
立即学习“C++免费学习笔记(深入)”;
public:
void add(int a, int b) { }
void add(double a, double b) { }
void add(int a, int b, int c) { }
};
这三个add函数构成重载。调用时根据参数自动匹配。
函数重写(覆盖,Override)
函数重写发生在基类与派生类之间,当派生类中定义了一个与基类虚函数同名、同参数列表、同返回类型的函数,并且基类中的函数被声明为virtual,此时派生类函数“覆盖”了基类函数。
重写是实现运行时多态的关键。通过基类指针或引用调用该函数时,会根据对象的实际类型动态调用对应的函数。
示例:class Base {
public:
virtual void show() { cout
};
class Derived : public Base {
public:
void show() override { cout
};
Base* ptr = new Derived();
ptr->show(); // 输出 Derived::show
函数隐藏(Hiding)
函数隐藏是指在派生类中定义了一个与基类同名的函数(不论参数是否相同),此时基类中所有同名函数都会被隐藏,即使参数列表不同也不会形成重载。
隐藏不要求函数是virtual,也不构成多态。如果未使用virtual,则调用哪个函数由指针/引用的静态类型决定。
示例:class Base {
public:
void func() { cout
void func(int x) { cout
};
class Derived : public Base {
public:
void func() { cout
};
Derived d;
d.func(10); // 编译错误!Base::func(int) 被隐藏
解决方法:在Derived中添加 using Base::func; 即可恢复重载。
关键区别总结
作用域不同:
- 重载:同一类或同一作用域内
- 重写:基类与派生类之间,有继承关系
- 隐藏:派生类屏蔽基类同名函数
是否需要virtual:
- 重载:不需要
- 重写:必须基类函数为virtual
- 隐藏:不需要virtual,但若基类是virtual则可能触发重写而非隐藏
调用机制:
- 重载:编译时确定(静态联编)
- 重写:运行时确定(动态联编)
- 隐藏:静态类型决定调用哪个函数
参数要求:
- 重载:参数列表必须不同
- 重写:参数列表必须完全相同
- 隐藏:只要名字相同即发生隐藏,无论参数
基本上就这些。掌握这三者的核心在于理解作用域、继承关系以及virtual的作用。重载是编译期行为,重写实现多态,隐藏则是C++名称查找规则的直接体现。










