C++中重载、覆盖与隐藏的区别(转)

  本文摘自林锐博士的《高质量C++/C编程指南》。

  成员函数的重载、覆盖(override)与隐藏很容易混淆,C++程序员必须要搞清楚概念,否则错误将防不胜防。

  1. 重载与覆盖  

  成员函数被重载的特征:
  (1)相同的范围(在同一个类中);
  (2)函数名字相同;
  (3)参数不同;
  (4)virtual关键字可有可无。
  覆盖是指派生类函数覆盖基类函数,特征是:
  (1)不同的范围(分别位于派生类与基类);
  (2)函数名字相同;
  (3)参数相同;
  (4)基类函数必须有virtual关键字。
  在下例中,函数Base::f(int)与Base::f(float)相互重载,而Base::g(void)被Derived::g(void)覆盖。

 1 #include <iostream>
 2
 3 class Base
 4 {
 5 public:
 6     void f(int x){ cout << "Base::f(int) " << x << endl; }
 7     void f(float x){ cout << "Base::f(float) " << x << endl; }
 8     virtual void g(void){ cout << "Base::g(void)" << endl; }
 9 };
10
11 class Derived : public Base
12 {
13 public:
14     virtual void g(void){ cout << "Derived::g(void)" << endl; }
15 };
16
17 void main(void)
18 {
19     Derived d;
20     Base *pb = &d;
21     pb->f(42); // Base::f(int) 42
22     pb->f(3.14f); // Base::f(float) 3.14
23     pb->g(); // Derived::g(void)
24 }

  2. 令人迷惑的隐藏规则

  本来仅仅区别重载与覆盖并不算困难,但是C++的隐藏规则使问题复杂性陡然增加。这里“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下:
  (1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆)。
  (2)如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。
  示例程序A中:
  (1)函数Derived::f(float)覆盖了Base::f(float)。
  (2)函数Derived::g(int)隐藏了Base::g(float),而不是重载。
  (3)函数Derived::h(float)隐藏了Base::h(float),而不是覆盖。

 1 // 示例A
 2
 3 #include <iostream>
 4
 5 class Base
 6 {
 7 public:
 8     virtual void f(float x){ cout << "Base::f(float) " << x << endl; }
 9     void g(float x){ cout << "Base::g(float) " << x << endl; }
10     void h(float x){ cout << "Base::h(float) " << x << endl; }
11 };
12
13 class Derived : public Base
14 {
15 public:
16     virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }
17     void g(int x){ cout << "Derived::g(int) " << x << endl; }
18     void h(float x){ cout << "Derived::h(float) " << x << endl; }
19 }; 

  据作者考察,很多C++程序员没有意识到有“隐藏”这回事。由于认识不够深刻,“隐藏”的发生可谓神出鬼没,常常产生令人迷惑的结果。
  示例B中,bp和dp指向同一地址,按理说运行结果应该是相同的,可事实并非这样。

 1 // 示例B
 2
 3 void main(void)
 4 {
 5     Derived d;
 6     Base *pb = &d;
 7     Derived *pd = &d;
 8
 9     // Good : behavior depends solely on type of the object
10     pb->f(3.14f); // Derived::f(float) 3.14
11     pd->f(3.14f); // Derived::f(float) 3.14
12
13     // Bad : behavior depends on type of the pointer
14     pb->g(3.14f); // Base::g(float) 3.14
15     pd->g(3.14f); // Derived::g(int) 3 (surprise!)
16
17     // Bad : behavior depends on type of the pointer
18     pb->h(3.14f); // Base::h(float) 3.14 (surprise!)
19     pd->h(3.14f); // Derived::h(float) 3.14
20 } 

  3. 摆脱隐藏

  隐藏规则引起了不少麻烦。示例8-2-3程序中,语句pd->f(10)的本意是想调用函数Base::f(int),但是Base::f(int)不幸被Derived::f(char *)隐藏了。由于数字10不能被隐式地转化为字符串,所以在编译时出错。

 1 class Base
 2 {
 3 public:
 4     void f(int x);
 5 };
 6
 7 class Derived : public Base
 8 {
 9 public:
10     void f(char *str);
11 };
12
13 void Test(void)
14 {
15     Derived *pd = new Derived;
16     pd->f(10); // error
17 }

从以上示例看来,隐藏规则似乎很愚蠢。但是隐藏规则至少有两个存在的理由:

  • 写语句pd->f(10)的人可能真的想调用Derived::f(char *)函数,只是他误将参数写错了。有了隐藏规则,编译器就可以明确指出错误,这未必不是好事。否则,编译器会静悄悄地将错就错,程序员将很难发现这个错误,流下祸根。
  • 假如类Derived有多个基类(多重继承),有时搞不清楚哪些基类定义了函数f。如果没有隐藏规则,那么pd->f(10)可能会调用一个出乎意料的基类函数f。尽管隐藏规则看起来不怎么有道理,但它的确能消灭这些意外。

  在以上示例中,如果语句pd->f(10)一定要调用函数Base::f(int),那么将类Derived修改为如下即可。

1 class Derived : public Base
2 {
3 public:
4     void f(char *str);
5     void f(int x) { Base::f(x); }
6 };
时间: 2024-10-19 02:01:09

C++中重载、覆盖与隐藏的区别(转)的相关文章

C++中成员函数的重载、覆盖和隐藏的区别

转载请注明,原创地址:点击打开链接   http://blog.csdn.net/u010587274/article/details/38928561 个人微信公众号:tanzi_888 (潭子技术圈) C++中成员函数的重载.覆盖和隐藏的区别: 1 重载(overload): 是函数名相同,参数列表不同 重载只是在类的内部存在.但是不能靠返回值类型来判断.1.1)相同的范围(在同一个类中)1.2)函数名字相同1.3)参数不同 1.4)Virtual关键字可有可无 2  覆盖 (overrid

java的重载、覆盖和隐藏的区别

重载:方法名相同,但参数不同的多个同名函数 注意:1.参数不同的意思是参数类型.参数个数.参数顺序至少有一个不同 2.返回值和异常以及访问修饰符,不能作为重载的条件(因为对于匿名调用,会出现歧义,eg:void a ()和int a() ,如果调用a(),出现歧义) 3.main方法也是可以被重载的 覆盖:子类重写父类的方法,要求方法名和参数类型完全一样(参数不能是子类),返回值和异常比父类小或者相同(即为父类的子类),访问修饰符比父类大或者相同 两同两小一大  注意:子类实例方法不能覆盖父类的

重载-重写(覆盖)-隐藏的区别

重载和重写搞的差不多了,隐藏有有点混乱了,在此记一下,留作后看 成员函数被重载的特征(1)相同的范围(在同一个类中):(2)函数名字相同:(3)参数不同:(4)virtual 关键字可有可无.重写(覆盖)是指派生类函数覆盖基类函数,--这是c++多态的标志,虚函数赋零则变成虚基类(纯粹的接口申明类),特征是(1)不同的范围(分别位于派生类与基类):(2)函数名字相同:(3)参数相同:(4)基类函数必须有virtual 关键字.“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下(1)如果派

C++重载、覆盖、隐藏的区别与举例

参考博客:http://blog.csdn.net/hexi_2000/article/details/4392107 //重载,覆盖,隐藏举例 #include <iostream> using namespace std; class A { public:     int n;     void fun()     {         cout<<"A::fun()"<<endl;     }     virtual void fun(int 

JAVA中方法和变量在继承中的覆盖和隐藏

我们知道,在JAVA中,子类可以继承父类,如果子类声明的方法与父类有重名的情况怎么办,大伙儿都知道要是重写,但是实际上这又分为两种情况,就是方法和变量在继承时的覆盖和隐藏问题. 隐藏 :child隐藏了parent的变量和方法,那么,child不能访问parent被隐藏的变量或者方法,但是,讲B转换成A中,可以访问A被隐藏的变量或者方法 覆盖 :child覆盖了parent的变量或者方法,那么,child不能访问parent被覆盖的变量或者方法,将child转换成parent后同样不能访问par

JAVA中方法和变量在继承中的覆盖和隐藏(一)

我们知道,在JAVA中,子类可以继承父类,如果子类声明的方法与父类有重名的情况怎么办,大伙儿都知道要是重写,但是实际上这又分为两种情况,就是方法和变量在继承时的覆盖和隐藏问题,这些概念性的东西看似无聊,但是在面试或者是SCJP认证题中围绕这些是会经常碰到的,所以这里来讨论下 首先我们来看几个概念 类变量:又叫静态变量,这种变量属于类,通过类名就可以访问类变量. 实例变量:属于类的实例,即对象,通过对象可以访问实例变量,但是不能通过类名访问实例变量. 静态方法:通过类名访问. 实例方法:对象访问,

Java中的覆盖和隐藏以及final关键字

Java覆盖和隐藏 (1) 变量只能被隐藏(包括静态和非静态),不能被覆盖: (2) 可以用子类的静态变量隐藏父类的静态变量,也可以用子类的非静态变量隐藏父类的静态变量,也可以用非最终变量(final)隐藏父类中的最终变量: (3) 静态方法(static)只能被隐藏,不能被覆盖: (4) 非静态方法可以被覆盖: (5) 不能用子类的静态方法隐藏父类中的非静态方法,否则编译会报错: (6) 不能用子类的非静态方法覆盖父类的静态方法,否则编译会报错: (7) 不能重写父类中的最终方法(final)

类成员函数的重载,覆盖和隐藏的区别

1.成员函数被重载的特征: 相同的范围,在同一个类中 函数名字相同 参数不同 virtual关键字可有可无 2.覆盖是指派生类函数覆盖基类函数,特征: 不同的范围(分别位于派生类和基类) 函数名字相同 参数相同 基类函数必须有virtual关键字 3.隐藏是指派生类的函数屏蔽了与其同名的基类函数,规则如下: a.如果派生类的函数与基类的函数同名,但是参数不同.此时,不论有无virtual关键字,基类的函数都将被隐藏(注意别与重载混淆) b.如果派生类的函数与基类的函数同名,并且参数也相同,但是基

重载、覆盖、隐藏的区别

重载:编译器根据函数不同的参数表,对同名函数的名称做修饰,然后这些同名函数就成了不同的函数.对于这两个函数的调用,在编译期间就已确定,是静态的.也就是说它们的地址在编译器就绑定了(早绑定),因此重载与多态无关. 覆盖(重写):子类重新定义父类虚函数的方法.当子类重新定义了父类的虚函数后,父类指针根据赋给它的不同的子类指针,动态的调用属于子类的该函数,这样的函数调用在编译期间是无法确定的.因此,这样的函数地址是在运行期间绑定的(晚绑定),重写与多态相关. 重载的特征       覆盖的特征 1.相