在派生类中重载基类函数

Posted

技术标签:

【中文标题】在派生类中重载基类函数【英文标题】:Overloading a base class function in the derived class 【发布时间】:2015-04-04 21:20:07 【问题描述】:

为什么如果派生类中重载了基类函数,则无法通过派生类的对象访问该函数的基类版本(即使是公共的)?

例如:

#include <iostream>
#include <string>

using namespace std;

class Base 
public:
    void f(int i) 
        cout << "\nInteger: " << i << endl;
    
;

class Derived : public Base 
public:
    void f(string s) 
        cout << "\nString: " << s << endl;
    
;


int main() 
    Base b;
    Derived d;
    //d.f(5);  Doesn't work
    d.f("Hello");
    //d.Base::f(5); works though
    return 0;

【问题讨论】:

This 可能是相关的 【参考方案1】:

名称查找在重载决议之前执行。名称查找在一个范围内开始,然后如果它没有找到该名称的声明,它会搜索一个封闭的范围,依此类推,直到找到该名称。在这种情况下,d.f 会找到声明 void Derived::f(string)。只有在Derived 中没有声明成员f 时,名称查找才会继续搜索基类。只有找到名称后,编译器才会确定是否存在适当的重载,如果存在,则哪个重载是最佳匹配。

请注意,您可以在派生类中重新声明基类函数,以强制找到它:

class Derived : public Base 
public:
    using Base::f;
    void f(string s) 
        cout << "\nString: " << s << endl;
    
;

现在名称查找将找到f 的两个重载,然后重载解析将确定调用哪一个。

【讨论】:

【参考方案2】:

如果你想在Derived::f之外还可以Base::f,可以加一行

using B::f;

Derived.

class Derived : public Base 
   public:
      using Base::f;
      void f(string s) 
         cout << "\nString: " << s << endl;
      
;

现在你可以使用了:

Derived d;
d.f(10);
d.f("Hello");

【讨论】:

以上是关于在派生类中重载基类函数的主要内容,如果未能解决你的问题,请参考以下文章

为啥派生类中的重写函数会隐藏基类的其他重载?

为啥派生类中的重写函数会隐藏基类的其他重载?

C++中派生类重写基类重载函数时需要注意的问题:派生类函数屏蔽基类中同名函数

派生类中的重载构造函数

派生类不从基类继承重载方法

从派生类调用重载函数