默认构造函数丢失 - 但我没有调用它?

时间:2013-01-08 16:11:23

标签: c++ constructor rule-of-three

我正在编写一个C ++应用程序,其中我有一个带有两个嵌套结构的Controller类,在我的头文件中定义如下:

class Controller {
    struct help_message {   // controller.hpp, line 19
        std::string summary;
        std::string details;
        help_message(const std::string&, const std::string&);
    };

    struct player_command {
        cmd_t cmd;
        help_message help;
        // cmd_t is my own typedef, irrelevant for this question
        player_command(const cmd_t&, const help_message&);
    };

    // more members...
};

在我的源文件中,我有这个:

Controller::player_command::player_command(const Controller::cmd_t& c, const help_message& h) {
    cmd = c;
    help = h;
};

Controller::help_message::help_message(const std::string& s, const std::string& d) {
    summary = s;
    details = d;
};

我觉得很好,但是当我编译时,这就是我得到的(controller.cpp第12行是上面源代码片段的第一行):

g++  -g -Wall -std=c++0x  -c -o controller.o controller.cpp
controller.cpp: In constructor ‘palla::Controller::player_command::player_command(void (palla::Controller::* const&)(const args_t&), const palla::Controller::help_message&)’:
controller.cpp:12:93: error: no matching function for call to ‘palla::Controller::help_message::help_message()’
controller.cpp:12:93: note: candidates are:
In file included from controller.cpp:7:0:
controller.hpp:22:3: note: palla::Controller::help_message::help_message(const string&, const string&)
controller.hpp:22:3: note:   candidate expects 2 arguments, 0 provided
controller.hpp:19:9: note: palla::Controller::help_message::help_message(const palla::Controller::help_message&)
controller.hpp:19:9: note:   candidate expects 1 argument, 0 provided
controller.hpp:19:9: note: palla::Controller::help_message::help_message(palla::Controller::help_message&&)
controller.hpp:19:9: note:   candidate expects 1 argument, 0 provided
make: *** [controller.o] Error 1

从我可以推断出,编译器在某处试图调用help_message的默认构造函数,它不存在。然后,它尝试将调用与我创建的构造函数以及生成的复制构造函数和赋值运算符进行匹配,并使每个参数的数量都失败。

但是我的代码的哪一部分是调用默认构造函数?我该如何解决这个错误?

3 个答案:

答案 0 :(得分:7)

player_command()构造函数首先默认构造help,然后分配给它:

Controller::player_command::player_command(const Controller::cmd_t& c, const help_message& h) {
    cmd = c;
    help = h;
};

将其更改为:

Controller::player_command::player_command(const Controller::cmd_t& c, const help_message& h)
:  cmd(c),
   help(h)
{
};

请参阅Benefits of Initialization lists

答案 1 :(得分:3)

player_command结构包含help_message(帮助),help_message没有默认构造函数。调用player_command构造函数时,默认情况下将为默认构造帮助成员变量。您正在立即为给定参数分配帮助,但这将在默认构造之后。而是将构造函数更改为:

Controller::player_command::player_command(const Controller::cmd_t& c, const help_message& h) : cmd(c), help(h) 
{}

这将调用cmd和help成员变量的复制构造函数,而不是执行默认构造,然后进行赋值。

答案 2 :(得分:0)

您没有使用避免复制构造函数的语法。参数通过引用传递(没有复制构造函数),但在分配给ivar时确实被复制了。

Class::Class(const Variable &var) {
  this->var = var; // assignment here, this invokes copy contructor!
}

您应该使用以下语法:

Class::Class(const Variable &var) : var(var) { }