对类层次的同名成员函数来说,有三种关系:重载(overload),覆盖(override)和隐藏(hide,oversee)。
1 成员函数的重载
只有在同一类定义中的同名成员函数才存在重载关系,主要特点是函数的参数类型和数目有所不同,但是不能出现函数参数的个数和类型均相同,仅仅依靠返回值类型不同来区分的函数,这和普通函数的重载是完全一致的。另外,重载和成员函数是否是虚函数无关。
例如:
class base
{
...
virtual int fun();
void fun(int);
void fun(double,double);
static int fun(char);
...
};
上述类定义中的4个fun函数便是重载关系。
成员函数被重载的特征:
1)相同的范围(在同一个类中)
2)相同的函数名字
3)不同的参数列表
4)virtual 关键字可有可无
2 成员函数的覆盖
覆盖是指:在派生类中覆盖基类中的同名函数,要求基类函数必须是虚函数,且:
1)与基类的虚函数有相同的参数个数
2)与基类的虚函数有相同的参数类型
3)与基类的虚函数有相同的返回类型:或者与基类虚函数的相同,或者都返回指针或引用,并且派生类虚函数所返回的指针或引用类型是基类中被替换的虚函数所返回的指针或引用类型的子类型或派生类型。
如下述代码中,B中的 fun1 覆盖了 A中的 fun1。
class A
{
public:
virtual void fun1(int,int){}
};
class B:public A
{
public:
void fun1(int,int){}
};
覆盖的特征为:
1)不同的范围 (分别位于派生类与基类中)
2)相同的函数名字
3)相同的参数
4)基类函数必须有virtual关键字
重载与覆盖的区别:
1)重载是同一个类中不同方法之间的关系,是水平关系;覆盖是子类和父类之间的关系,是垂直关系
2)覆盖要求参数列表相同,重载要求参数列表不同;覆盖要求返回类型相同,重载则不要求
3)覆盖关系中,调用方法体是根据对象的类型来决定的,重载关系是根据调用时的实参表与形参表来选择方法体的。
3 成员函数的隐藏
隐藏指的是在某些情况下,派生类中的函数屏蔽了基类中的同名函数,这些情况包括:
1)两个函数参数相同,但基类函数不是虚函数。和覆盖的区别在于基类函数是否是虚函数。如下述代码:
class A
{
public:
void fun(int xp)
{
cout<<xp<<endl;
}
};
class B:public A
{
public:
void fun(int xp){}
};
上述代码若有:
B b;
b.fun(2);
则调用的将是B中的函数 fun,若需调用A中的函数fun,可以以这种形式调用;
B b;
b.A::fun(2);
2) 两个函数的参数列表不同,无论基类函数是否是虚函数,基类函数都将会被屏蔽。和重载的区别在于两个函数不在同一个类中。如下述代码:
class A
{
public:
virtual void fun(int xp)
{
cout<<xp<<endl;
}
};
class B:public A
{
public:
void fun(char* xp){}
};
在上述代码中,若:
B b;
b.fun(2);
则是错误的,这是因为A中的函数 fun 被隐藏了,此时编译器找不到参数为int型的fun函数。此时可通过:
B b;
b.A::fun(2);
调用A中的fun函数。