C++复制构造函数行为
C++ copy constructor behaviour
C++代码中有一部分我不太明白。我也不知道我应该去哪里搜索有关它的信息,所以我决定问一个问题。
#include <iostream>
#include <string>
using namespace std;
class Test
{
public:
Test();
Test(Test const & src);
Test& operator=(const Test& rhs);
Test test();
int x;
};
Test::Test()
{
cout << "Constructor has been called" << endl;
}
Test::Test(Test const & src)
{
cout << "Copy constructor has been called" << endl;
}
Test& Test::operator=(const Test& rhs)
{
cout << "Assignment operator" << endl;
}
Test Test::test()
{
return Test();
}
int main()
{
Test a;
Test b = a.test();
return 0;
}
为什么我得到的输入是
Constructor has been called
Constructor has been called
?a.test() 通过调用 "Test()" 创建一个新实例,这就是显示第二条消息的原因。但是为什么没有调用复制构造函数或赋值呢?如果我将"return Test()"更改为"return *(new Test())",则调用复制构造函数。
那么为什么不叫第一次呢?
编译器非常聪明。两个副本 - 从test
返回并初始化b
(不是这不是赋值) - 根据以下规则(C++11 §12.8)省略:
当尚未绑定到引用 (12.2) 的临时类对象将被复制/移动到具有相同 CV-UNQUALIFIED 类型的类对象时,可以通过将临时对象直接构造到省略的复制/移动的目标中来省略复制/移动操作
编译器可以这样做,即使它会改变程序的行为(例如删除输出消息)。应不要编写具有其他副作用的复制/移动构造函数和赋值运算符。
请注意,这只是可能发生复制省略的四种情况之一(不包括 as-if 规则)。
对 a.test() 的调用按值返回,然后将该值分配给 b "复制"返回值。 这将调用复制构造函数。
相关文章:
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 为什么在C++中使用私有复制构造函数与删除复制构造函数
- 当从函数参数中的临时值调用复制构造函数时
- 如果有一个模板构造函数只有一个泛型参数,为什么我必须有一个复制构造函数
- 使用复制构造函数复制双精度数组
- C 无可行的构造函数复制类型的变量
- 没有可行的构造函数复制类型 'MyString' 的数组元素
- 编译时,复制构造函数/复制分配和正常功能调用优化之间是否存在任何区别
- 如何最小化调用列表构造函数(复制构造函数)的次数?
- C 11矢量构造函数复制与范围
- 我定义了一个非复制构造函数;复制构造函数还会被隐式定义吗
- 可以将构造函数复制为转换运算符
- 将基类指针的构造函数复制到子类
- C++树类:构造函数/复制/内存泄漏
- 如何制作这个在模板构造函数复制中使用类型定义的类型的模板
- 将构造函数复制为模板化的成员函数
- 绕过私有复制构造函数/复制赋值C++
- C++通过构造函数复制对象
- 复制构造函数 - 复制C++中的对象
- 将带unique_ptr的类的构造函数复制到作为成员的抽象类