编译器找不到在基类中实现的虚函数

Posted

技术标签:

【中文标题】编译器找不到在基类中实现的虚函数【英文标题】:Virtual function implemented in base class not being found by compiler 【发布时间】:2011-04-10 08:13:33 【问题描述】:

我遇到一种情况,编译器似乎没有找到与另一个成员函数同名的虚函数的基类定义/实现。

struct One ;

struct Two ;

struct Base

    virtual void func( One & );
    virtual void func( Two & ) = 0;
;

struct Derived : public Base

    virtual void func( Two & );
;

void Base::func( One & )


void Derived::func( Two & )


// elsewhere
void this_fails_to_compile()

    One one;
    Derived d;
    d.func( one );

我使用的是 Visual C++ 2008。错误信息是:

错误 C2664:“Derived::func”:无法将参数 1 从“一”转换为“二 &”

我原以为基于类型的调度会起作用并调用定义的基类函数。如果我添加一个Derived::func( One & ),它会编译并正确调用,但在我的情况下,该版本的函数可以在基类中完成,通常派生类不需要自己实现它。我目前正在通过在基类中放置一个不同名称的非虚拟函数来解决它,该函数将调用转发给导致问题的函数:

// not virtual, although I don't think that matters
void Base::work_around( One & one )

    func( one );

这可行,但显然不太理想。

我在这里缺少什么继承和/或名称隐藏规则?

【问题讨论】:

补充一下,g++也会产生类似的错误信息。 【参考方案1】:

您将方法隐藏在派生类中。最简单的解决方案是在派生类中添加 using 声明。

struct Derived : public Base

    using Base::func;
    virtual void func( Two & );
;

问题在于,当编译器尝试在调用 d.func(one) 中查找 func 标识符时,它必须从 Derived 向上执行此操作,但它会在找到 func 的第一个上下文中停止标识符,在本例中为Derived::func。没有执行进一步的查找,编译器只看到Derived::func( Two& )

通过添加using Base::func; 指令,当编译器看到Derived 定义时,它会将所有Base::func 声明带入作用域,它会发现Base::func( One & ) 没有在Derived 中被覆盖.

另请注意,如果您通过对 Base 的引用进行调用,那么编译器将找到 func 的两个重载,并将每个重载适当地分派给最终的覆盖器。

Derived d;
Base & b = d;
b.func( one ); // ok even without the 'using Base::func;' directive

【讨论】:

啊。我没有意识到它会在找到名称匹配后立即停止搜索,即使参数不匹配。感谢您快速而彻底的回答。 @Kurt:这是名称隐藏的问题。很烦人。【参考方案2】:

您将func(One&) 函数隐藏在Derived 中。您可以使用完全限定名称:

d.Base::func( one );

【讨论】:

一般来说,使用完全限定名称可能不是最好的主意。考虑是否将新的Derived2 类添加到将覆盖Base::func(One&) 的层次结构中。现在,如果通过引用 Derived 来持有 Derived2 的实例并与上面的代码一起使用,它将调用 Base::func(One&) 而不是最终的覆盖器 Derived2::func(One&) 我什至不知道您可以指定成员函数的限定名称!不过语法看起来很糟糕。

以上是关于编译器找不到在基类中实现的虚函数的主要内容,如果未能解决你的问题,请参考以下文章

C++ 虚函数(virtual) 和纯虚函数(pure virtual) 的区别

是否可以通过在基类中添加新的虚函数来破坏代码?

虚函数

抽象类

C++ 纯虚函数

为啥“基类对象”不能调用它自己的虚函数? C++