ama*_*jxq 6 c++ gcc copy-constructor c++11
我使用下面的代码来测试copy elision:
class foo
{
public:
foo() {cout<<"ctor"<<endl;};
foo(const foo &rhs) {cout<<"copy ctor"<<endl;}
};
int g(foo a)
{
return 0;
}
int main()
{
foo a;
g(std::move(a));
return 0;
}
Run Code Online (Sandbox Code Playgroud)
我预计只会调用默认构造函数,因为参数g()
是一个右值,副本将被省略.但结果显示默认构造函数和复制构造函数都被调用.为什么?
如果我将函数调用更改为g(foo())
,则复制将被删除.foo()
和的返回类型有什么区别std::move(a)
?如何在左值上编译编译器?
复制省略只能在少数特定情况下发生,其中最常见的是复制临时(其他人正在返回本地,以及抛出/捕获异常).您的代码没有临时生成,因此不会删除任何副本.
拷贝构造函数被调用,因为foo
没有一个移动构造函数(move构造函数不含蓄与明确的拷贝构造函数的类产生的),所以 std::move(a)
相匹配的foo(const foo &rhs)
构造函数(这是用来构造函数参数).
在以下情况下可以省略左值的副本(尽管无法强制编译器执行省略):
foo fn() {
foo localAutomaticVariable;
return localAutomaticVariable; //Copy to construct return value may be elided
}
int main() {
try {
foo localVariable;
throw localVariable; //The copy to construct the exception may be elided
}
catch(...) {}
}
Run Code Online (Sandbox Code Playgroud)
如果要在传递函数参数时避免复制,可以使用移动构造函数来传递给定对象的资源:
class bar {
public:
bar() {cout<<"ctor"<<endl;};
bar(const bar &rhs) {cout<<"copy ctor"<<endl;}
bar(bar &&rhs) {cout<<"move ctor"<<endl;}
};
void fn(bar a)
{
}
//Prints:
//"ctor"
//"move ctor"
int main()
{
bar b;
f(std::move(b));
}
Run Code Online (Sandbox Code Playgroud)
此外,每当允许复制省略但未发生复制省略时,将使用移动构造函数(如果可用).
归档时间: |
|
查看次数: |
2334 次 |
最近记录: |