关于复制构造

About copy construct

本文关键字:复制 于复制      更新时间:2023-10-16
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)"并且会出错,为什么呢?

构造函数的函数体(左大括号和右大括号之间的部分)与任何其他函数的主体没有什么不同。 您是否希望编译:

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

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

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

: price(orig.price)

等效于这样的语句:

std::string price(orig.price);

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

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

price = orig.price;

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

这是

不正确的,因为价格已经构建好了。如果它已经构造,则不能调用它的复制构造函数。

你必须做一些类似 price = orig.price 的事情;

检查编译错误,你就会明白为什么。在成员初始值设定项列表中,price(orig.price)是使用值 orig.price price直接初始化成员变量。在复制构造函数的主体中,price(orig.price)是对接受std::string的重载operator()的调用 std::string 。由于没有这样的重载,因此会出现编译错误。