C++覆盖,隐藏,重载

Posted unknowcodemaker

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++覆盖,隐藏,重载相关的知识,希望对你有一定的参考价值。

 

定义

  • 函数覆盖
    (1) 作用域不同
    (2) 有virtual关键字
    (3) 参数列表/返回值/调用约定必须相同
  • 函数隐藏
    (1) 作用域不同
    (2) 函数名相同
    (3) 参数列表/返回值/调用约定不考虑
  • 函数重载
    (1) 作用域相同
    (2) 函数名相同
    (3) 参数列表不同,返回值/调用约定不考

练习

class Base

public:
    virtual void Handle1(float x)
    
        cout << "Base::Handle1(float) " << x << endl;
    
    void Handle2(float x)
    
        cout << "Base::Handle2(float) " << x << endl;
    
    void Handle3(float x)
    
        cout << "Base::Handle3(float) " << x << endl;
    
;

class Derived : public Base

public:
    virtual void Handle1(float x)
    
        cout << "Derived::Handle1(float) " << x << endl;
    
    void Handle2(int x)
    
        cout << "Derived::Handle2(int) " << x << endl;
    
    void Handle3(float x)
    
        cout << "Derived::Handle3(float) " << x << endl;
    
    void Handle3(double x)
    
        cout << "Derived::Handle3(double) " << x << endl;
    
;

观察以上类声明及实现,试预测以下main中的执行结果,注意两点:

  1. 调用的方法是哪个?
  2. 调用造成的原因,是重载、覆盖、隐藏中的哪一种?
 void main(void)
 
     Derived DervObj;
     Base *pBase = &DervObj;   
     Derived *pDerv = &DervObj;
 
     pBase->Handle1(3.14f);
     /******************************************
     预测:
           输出Derived::Handle1(float)3.14
           是间接调用,覆盖
     运行结果:Derived::Handle1(float)3.14
     ******************************************/
 
 
     pDerv->Handle1(3.14f);
     cout << endl;
     /******************************************
     预测:
           输出Derived::Handle1(float)3.14
           是间接调用,隐藏
     运行结果:Derived::Handle1(float)3.14
     ******************************************/
 
 
     pBase->Handle2(3.14f);
     /*****************************************
     预测:
           输出Base::Handle2(float)3.14
           是直接调用,隐藏
     运行结果:Base::Handle2(float)3.14
     *****************************************/
 
 
 
     pDerv->Handle2(3.14f);
     /****************************************
     预测:
           输出Derived::Handle2(int)3
           是直接调用,隐藏
     运行结果:Derived::Handle2(int)3
     *****************************************/
     cout << endl;
 
 
 
     pBase->Handle3(3.14f);
     /****************************************
     预测:
           输出Base::Handle3(float)3.14
           直接调用,隐藏
     结果:Base::Handle3(float) 3.14
     
     ****************************************/
 
 
     pDerv->Handle3(3.14f);
     /***************************************
       预测:输出Derived::Handle3(float)3.14
             直接调用,隐藏
       结果:Derived::Handle3(float) 3.14
     
     ***************************************/
 
 
     pDerv->Handle3(3.14);
     /**************************************
       预测:输出Derived::Handle3(double)3.14
             直接调用,隐藏
       结果:Derived::Handle3(double) 3.14
     **************************************/
     cout << endl;
     
     pDerv->Handle3(3);
     /****************************************
       Handle3没有对应的整数形参,只有float和double
       3既可以转换为float,又可以转换为double,所以编译时报错。
     /****************************************/
     system("pause");
 

总结

当覆盖,隐藏,重载同时存在时,如何确定调用了某个类的某个函数?
a)查看调用该函数的数据类型,并找到该类型的作用域(函数隐藏)
b)在该作用域中找到同名函数,满足参数列表相同(可以做适当的类型转换),在从这些同名函数中选出最佳的,
如果有多个则产生二义性(函数重载)
c)查看选出的最佳匹配函数是否是虚函数,如果是虚函数并且调用该虚函数的数据类型为指针或引用,
那么就是间接调用,否则就是直接调用(函数覆盖)

 

以上是关于C++覆盖,隐藏,重载的主要内容,如果未能解决你的问题,请参考以下文章

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

C++覆盖,隐藏,重载

C++学习(四五零)重载覆盖隐藏

C++之重载覆盖和隐藏

C++重载隐藏和覆盖的区别

类中的同名函数关系,重载,覆盖/重写,隐藏