首页 > 解决方案 > 未见类外函数重载

问题描述

考虑以下代码片段:

enum class Bar {
    A
};

void foo(Bar) {}

struct Baz {
    void foo() {
        foo(Bar::A);
    }
};

它无法编译,来自 gcc 9.2 的消息是:

:12:19: error: no matching function for call to 'Baz::foo(Bar)'
 12 |         foo(Bar::A);
    |              

我不怀疑这是一个错误,因为 clang 10 也失败了。关于这种情况,我有两个问题:

  1. 标准在哪里为此类重载定义了行为?

  2. 以这种方式指定编译器行为的可能原因是什么?

活生生的例子

标签: c++overloadinglanguage-lawyername-lookup

解决方案


foo对inside的调用Baz::foo()只会在类中查找名称。如果您的意思是foo在 class 之外使用声明Baz,则需要使用范围解析运算符,如下所示:

enum class Bar {
    A
};

void foo(Bar) {}

struct Baz {
    void foo() {
        ::foo(Bar::A);   // looks up global 'foo'
    }
};

请注意,无作用域调用foo失败,因为Bar::foo在最近的作用域中找到了一个。如果您对函数命名不同,则在 中找不到函数Bar,编译器将在函数的外部范围内查找。

enum class Bar {
    A
};

void foo(Bar) {}

struct Baz {
    void goo() {   // not 'foo'
        foo(Bar::A);  // this is fine, since there is no 'Bar::foo' to find
    }
};

这是cppreference类定义的引用。

e) 如果此类是命名空间的成员,或者嵌套在作为命名空间成员的类中,或者是作为命名空间成员的函数中的本地类,则搜索命名空间的范围直到类、封闭类或函数的定义。如果查找由朋友声明引入的名称:在这种情况下,只考虑最里面的封闭命名空间,否则查找将继续封闭命名空间,直到像往常一样进入全局范围。

当然,这仅适用于类定义,但对于成员函数(这是您的示例),它说

对于在成员函数体内使用的名称、成员函数的默认参数、成员函数的异常规范或默认成员初始值设定项,搜索的范围与 [类定义] 中的相同,...

所以同样的逻辑也适用。


推荐阅读