0
class Wood {
public:
    Wood();
    Wood(const Wood&); //copy constructor
    ~Wood();

private:
    string price;
};

Wood::Wood(const Wood& orig) {
    price(orig.price);   **//error, why?**
}

Wood::Wood(const Wood& orig) : price(orig.price) { //rigth

}

如果我使用构造初始化并且它是正确的。但是如果使用“price(orig.price)”会出错,为什么?

4

3 回答 3

4

构造函数的函数体(左括号和右括号之间的部分)与任何其他函数的函数体没有什么不同。你会期望这个编译:

std::string a, b;
a(b);  // <--- this line?

不,当然不是。为了编译它,std::string需要类似operator()重载的东西,它需要另一个字符串。它没有那个。

初始化列表中的代码不同。初始化列表中的表达式不会被解释为普通语句,就像函数体内的那些。它们被解释为初始化(例如构造函数调用)。因此,在初始化列表中,这是:

: price(orig.price)

相当于这样的语句:

std::string price(orig.price);

除了price不需要指定的类型,因为这已经在类定义中完成了。

请注意,您不能在构造函数主体内进行成员初始化,因为当您到达那里时,所有成员都已初始化。这就是您需要初始化列表的原因。当然,您可以在构造函数主体中进行赋值:

price = orig.price;

但这与初始化不同。它不适用于某些类型(例如 const 成员、引用成员或没有默认构造函数的成员)。对于某些类型,它的效率可能会降低,因为您首先构造(使用默认构造函数),然后进行分配。但是对于许多类型来说,这并不重要,因为默认构造几乎没有任何成本。

于 2015-01-30T07:56:13.083 回答
1

这是不正确的,因为价格已经构建。如果它已经构建,则不能将其称为复制构造函数。

您必须执行价格 = orig.price; 之类的操作。

于 2015-01-30T07:52:36.693 回答
1

检查编译错误,您将了解原因。在成员初始化器列表中,price(orig.price)直接price使用 value初始化成员变量orig.price。在复制构造函数的主体中,是对接受 aprice(orig.price)的重载的operator()调用。由于没有这样的重载,你会得到一个编译错误。std::stringstd::string

于 2015-01-30T07:54:44.370 回答