继承中的重载,重写及相应隐藏规则

来源:互联网 发布:淘宝店铺教程视频 编辑:程序博客网 时间:2024/06/05 16:27

        在C++中的继承中写函数的时候,我们问题会遇到重载,重写,隐藏函数这三大问题。在面向对象的程序设计中,对这三个概念的理解相应重要,否则会出现一些莫明其妙的事件。下面我们来解释下这三个的概念:

重载:即在同一个类中声明相同的函数名,但型参列表的参数类型与及参数的数目不同时,应会发生重载。

示例代码:

class A{public://函数名相同,形成的重载void f(){};void f(int a){};void f(double b){};};

重写:即基类中声明virual的函数,子类继承相应的函数,但实现了自己的方法。我们就称为重写(也称覆盖)即子类调用同名的函数时,调用的是从基类重写的函数,而还在是基类的函数实现。重写是多态产生的机制,把基类中带有virual关键字的函数,在子类中实现自己的版本。这叫重写。注意与隐藏的区别之处。

示例代码:

class A{public:virtual void print(){cout<<"A print()"<<endl;}};class B : public A{public:virtual void print(){//此时就是重写,实现自己的处理cout<<"B print()"<<endl;}};
此时,调用B对象的print方法时,就会执行B对象改造后的print函数版本。

隐藏:这个在C++中十分特殊,正是这条规则的存在,使得这三个概念的难度一下子就提高了很多。具有有下面有种应用场合。

1.对基类中函数不存在virual关键字,子类继承后,如果再次定义一个同名函数(此时注意不是重载),不管定义的同名函数形参列表类型及个数是否相同,基类的网名函数彻底被隐藏。此时调用子类对象中的函数时,就只能调用子类中定义的函数(注意此函数并不继承自基类的网名函数,并不是所谓的重写)。

class A{public: void print(string s){cout<<"A print()"<<endl;}};class B : public A{public:void print(int x){//此时会发生隐藏,没有重载cout<<"B print(int a)"<<endl;}};int main(){B b;b.print("hello");//error,参数类型错误。此时调用的是print(int x)函数b.print(1);        //right,此时基类同名函数被隐藏}
下面的是基类带有virual关键字代码示例,此时子类重写了基类中继承的virual同名函数,然后有又定义也自己和基类同名但参数类型不同的函数。

代码示例

class A{public: virtual void print(string s){cout<<"A print()"<<endl;}};class B : public A{public:virtual void print(string s){//函数重写cout<<"B print(string s)"<<endl;}void print(int x){//此时会发生隐藏,没有重载cout<<"B print(int a)"<<endl;}};int main(){B b;b.print("hello");  //right,->B print(string s) //被重写的函数没有被隐藏b.print(1);        //right, ->B print(int a) }
注意:此时调用的b.print("hello")正确执行。但如果子类没有重写基类当中的同名virual函数。则此时b.print("hello")不能成功。发现所谓的隐藏现象。凡是基类中virual函数,没有在子类中重写的,而子类有定义也自己的同名不同参数类型的函数时,所有从基类继承的同名函数都被隐藏,只有自己定义的函数可见。看下面的代码示例:就会出现错误情况

class A{public: virtual void print(string s){cout<<"A print()"<<endl;}};class B : public A{public://virtual void print(string s){//函数重写//cout<<"B print(string s)"<<endl;//}注释后,就出现隐藏现象了,一定要留意此规则void print(int x){//此时会发生隐藏,没有重载cout<<"B print(int a)"<<endl;}};int main(){B b;b.print("hello");  //error,此时b类中没有print(string s)类型的函数,被我们定义的同名函数隐藏了b.print(1);        //right, ->B print(int a) }


原创粉丝点击