C ++编译器错误C2751 - 究竟是什么导致它?

时间:2017-05-18 10:28:20

标签: c++ visual-c++ constructor function-parameter

我正在努力解决C2751编译器错误,并且不太明白究竟是什么原因造成的。以下小代码会产生错误:

#include <iostream>  

class A {
public:
    A () { std::cout << "A constructed" << std::endl; };
    static A giveA () { return A (); }
};

class  B {
public:
    B (const A& a) { std::cout << "B constructed" << std::endl; }
};


int main () {

    B b1 = B (A::giveA ()); // works
    B b2 (B (A::giveA ())); // C2751
    B b3 (A::giveA ()); // works

}

编译器输出:

consoleapplication1.cpp(21): error C2751: 'A::giveA': the name of a function parameter cannot be qualified

为什么我无法为b2显式调用构造函数?

4 个答案:

答案 0 :(得分:6)

这是most vexing parse的问题。在clang下编译可以提供完整的诊断:

<source>:18:17: error: parameter declarator cannot be qualified
    B b2 (B (A::giveA ())); // C2751
             ~~~^

<source>:18:10: warning: parentheses were disambiguated as a function declaration [-Wvexing-parse]
    B b2 (B (A::giveA ())); // C2751
         ^~~~~~~~~~~~~~~~~

<source>:18:11: note: add a pair of parentheses to declare a variable
    B b2 (B (A::giveA ())); // C2751
          ^
          (              )
1 warning and 1 error generated.
Compiler exited with result code 1

正如编译器建议的那样修复它:

B b2 ((B (A::giveA ()))); // no error 

答案 1 :(得分:2)

该特定指令含糊不清;你说的方式可能是函数声明或变量定义。

使用额外的括号应该有帮助 - 它是明确的,因为你不能声明/调用带有双括号的函数:

B b1 = B( A::giveA () ); // works
B b2(( B( A::giveA() ) )); // works for me
B b3( A::giveA() ); // works

另见:https://en.wikipedia.org/wiki/Most_vexing_parse

答案 2 :(得分:2)

B b2 (B (A::giveA ()));

即使您有B类的复制构造函数,您仍然会收到此错误,因为上面的行符合函数声明的语法规则。括号B(A::give())左右可以解决您的问题。

答案 3 :(得分:2)

啊,most vexing parse的问题。

这一行

B b2 (B (A::giveA ())); // C2751

被解释为函数b2,返回B并获取函数类型A::giveA的名为B()的参数。 C2751错误是由无效名称A::giveA引起的:您不能使用限定名称作为参数名称。

要将临时B对象复制到b2(我相信您的意图),您可以添加更多括号

B b2 (( B (A::giveA ()) ));

或使用大括号(只要没有接受包含可转换为B类型的类型的初始化列表的构造函数)

B b2 {B (A::giveA())};