首页 > 学院 > 开发设计 > 正文

重载、覆盖、隐藏的区别

2019-11-10 17:57:50
字体:
来源:转载
供稿:网友

学习过C++基础课程的人,对重载函数的概念相对熟悉一些,但是对覆盖和隐藏就相对陌生了,在编程技术的学习上,你会发现一个规律,很多知识点你都知道,但是一旦用起来,就常常出错,为什么呢?笔者一直都在探索这个问题。。

重载函数有哪些特性呢?

   《高质量C++/C编程指南》已经清晰的列出了重载函数的特性:

    (1)相同的范围(在同一个类中);    (2)函数名字相同;    (3)参数不同;    (4)virtual关键字可有可无。

  因为函数参数不同,可以简单的理解为:两个重载函数是不同的函数,调用者能够明确的根据不同的参数来调用不同的函数。那么如果存在这样两个函数,编译器怎么处理呢?

 

class A{public:    void Func(int a, int b=0) {PRintf("This is Func1/n");}    void Func(int a) {printf("This is Func2/n");}};

int main(){    A a;    a.Func(5);    return 0;}

 

当然,对于这样两个函数,调用者不知道应该调用哪个函数,故编译器直接报错。

我们在看看,覆盖和隐藏分别是什么特性呢?从字面意思来讲,覆盖和隐藏都具有一个把另一个给遮住了,那只不过是谁遮谁的问题。

覆盖:是指派生类函数覆盖基类函数,只作用于派生类函数,其特性为:(1)不同的范围(分别位于派生类与基类);(2)函数名字相同;(3)参数相同;(4)基类函数必须有virtual关键字。

我们发现,这里用到了虚函数,实际上虚函数的作用,就是实现覆盖。

隐藏:是指派生累函数将基类函数给藏起来了,当然只作用于派生类函数,其特性与覆

盖不同。(1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏。

(2)如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏。

这里的隐藏和覆盖的含义,感觉上非常的模糊,甚至不清,并且隐藏的规则,常常令人感觉到神出鬼没。。。

请参看如下实例:

class Base{public: virtual void f(float x){ cout << "Base::f(float) " << x << endl; } void g(float x){ cout << "Base::g(float) " << x << endl; } void h(float x){ cout << "Base::h(float) " << x << endl; }};

class Derived : public Base{public: virtual void f(float x){ cout << "Derived::f(float) " << x << endl; } void g(int x){ cout << "Derived::g(int) " << x << endl; } void h(float x){ cout << "Derived::h(float) " << x << endl; }};

void main(void){ Derived  d; Base *pb = &d; Derived *pd = &d; pb->f(3.14f);  pd->f(3.14f);   // Bad : behavior depends on type of the pointer pb->g(3.14f);  pd->g(3.14f);   // Bad : behavior depends on type of the pointer pb->h(3.14f);  pd->h(3.14f); }

 

令人非常不解,bp和dp指向同一地址,按理说运行结果应该是相同的,可事实并非这样。所以我们还是要想办法摆脱隐藏。

还有一点要切忌,对于static这种静态成员函数,是属于类的方法,而不是对象的方法,所以static方法绝对不能被覆盖或者隐藏。

转自:http://blog.csdn.net/hexi_2000/article/details/4392107


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表