C++中函数重载、隐藏、重写的区别
1.函数重载
1.1定义
C++规定在同一作用域中,同名函数的形式参数(指参数的个数、类型或者顺序)不同时,构成函数重载。
2.1用法
例:
namespace hello
{
int add(int a,int b)
{
return a + b;
}
double add(double a,double b)
{
return a + b;
}
}
1.3注意:
(1)函数返回值类型与构成函数重载没有任何关系;
(2)类的静态成员函数与普通成员函数可以形成重载;
(3)必须在同一作用域下才能形成函数重载,像类成员函数之间的重载、全局函数之间的重载。
2.同名隐藏(也叫重定义)
2.1定义
同名隐藏指不同作用域中定义的同名函数构成函数隐藏(不要求函数返回值和函数参数类型相同)。比如派生类成员函数屏蔽与其同名的基类成员函数、类成员函数屏蔽全局外部函数。请注意,如果在派生类中存在与基类虚函数同返回值、同名且同形参的函数,则构成函数重写。
2.2注意事项
1.在不同作用域;
2.函数名相同即可;(不要求函数返回值,参数等相同。)
3.函数重载发生在同一作用域,函数隐藏发生在不同作用域。
4.派生类成员函数与基类成员函数同名但参数不同。此时基类成员函数将被隐藏(注意别与重载混淆,重载发生在同一个类中);
2.3用法
例
class A
{
void add()
{
cout << "add function" << endl;
}
void fun1()
{
cout << "你好" << endl;
}
}
class B:public A
{
void add()
{
cout << "add function" << endl;
}
void fun1()
{
cout << "你好" << endl;
}
}
int main()
{
A a;
a.add();
a.fun1();
B b;
b.A::add();a的add函数被隐藏,要访问就用类名加作用域的方法。
b.A::fun1();
return 0;
}
3.函数重写(也叫函数覆盖)
3.1定义
派生类中与基类同返回值类型、同名和同参数的虚函数重定义,构成虚函数覆盖,也叫虚函数重写。
也可以说成是派生类中与基类同返回值类型、同名和同参数的虚函数重定义。
注:关于返回值类型存在一种特殊情况,即协变返回类型(covariant return type)。
如果虚函数函数返回指针或者引用时,子类中重写的函数返回的子类的指针或者引用,父类中被重写函数所返回父类的指针或引用的类型(这就是所谓的协变返回类型。)
3.2注意
1.要在派生类和基类;
2.虚函数重定义;
3.返回值类型(有协变这种特殊情况。)
3.3用法
例
#include <iostream>
using namespace std;
class A{
public:
virtual void show(){
cout<<"A"<<endl;
}
};
class B:public A{
public:
virtual void show(){
cout<<"B"<<endl;
}
};
int main(){
A a;
a.show();
B b;
b.show(); //对函数show()的实调用
b.A::show(); //对函数show()的实调用
A *pa = NULL;
pa = &b;
pa->show(); //对函数show()的虚调用
pa->A::show(); //对函数show()的实调用
}
程序最后运行结果:
A
B
B
A
4.总结
(1)函数重载发生在相同作用域;
(2)函数隐藏发生在不同作用域;
(3)函数覆盖就是函数重写。准确地叫作虚函数覆盖和虚函数重写。
可以这样来判断:
1.函数重载:有相同的作用域,可以有(virtual)这个关键字,也可以没有这个关键字,但它们的函数名必须相同,形参列表不同,返回值可以相同,也可以不同。
2.同名隐藏:作用域不同,可以有(virtual)这个关键字,也可以没有这个关键字,但它们的函数名必须相同,形参列表可以相同,也可以不同,返回值可以相同,也可以不同。
3.函数重写:作用域不同,有(virtual)这个关键字,但它们的函数名必须相同,形参列表必须相同,返回值也相同,但要注意协变这种特殊情况。