#include <iostream>
using namespace std;
class A
{
int x;
public:
A(int c) : x(c) {}
A(const A& a) { x = a.x; cout << "copy constructor called" << endl;}
};
class B
{
A a;
public:
B(int c) : a(c) {}
B(const B& b) : a(b.a) { }
A get_A() { return a;}
};
int main()
{
B b(10);
A a1 = b.get_A();
}
在上面的代码中,我预计'复制构造函数调用'消息会弹出两次,因为首先,b.get_A()将创建一个临时对象,调用复制构造函数(1),第二,它将复制其引用到a1的拷贝构造函数(2),从而显示两条消息。
但是,代码实际上会产生一个名为'copy'的复制构造函数。为什么?
答案 0 :(得分:8)
在某些情况下,C ++标准允许copy constructor to be elided。通常,这意味着是否将从临时变量复制构造对象。它可以在适当的位置构建。
在这种情况下,get_A();
已将副本返回到临时副本中。然后,您将a1
分配给该临时变量。允许编译器忽略额外的副本并使用返回值a1
构建get_A()
。
即使复制构造函数有副作用,也可能发生此优化。
复制省略是唯一允许改变可观察副作用的优化形式。因为某些编译器不允许在允许的每种情况下执行复制省略,所以依赖于复制/移动构造函数和析构函数的副作用的程序是不可移植的。
答案 1 :(得分:3)
在C ++ 11中,代码可能会调用移动构造函数来移动对象而不是复制它,或者在NRVO形式的C ++ 03和C ++ 11编译器优化中都可以应用 {{3 }} 来删除副本。
请注意,后者( copy elision )取决于编译器的功能,并且无法保证在C ++ 11中保证前者( Move语义)。