コピーの省略は、いくつかの特定の状況でのみ発生する可能性があります。最も一般的なのは、一時的なコピーです (他の状況では、ローカルが返され、例外がスロー/キャッチされます)。コードによって生成される一時的なものはないため、コピーは省略されません。
コピー コンストラクターが呼び出されfoo
ているのは、移動コンストラクターがないため (移動コンストラクターは、明示的なコピー コンストラクターを持つクラスに対して暗黙的に生成されないため)、 コンストラクター (関数引数の構築に使用される) と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(...) {}
}
関数の引数を渡すときにコピーを避けたい場合は、渡されたオブジェクトのリソースを盗む移動コンストラクターを使用できます。
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));
}
また、コピーの省略が許可されているが発生しない場合は常に、使用可能な場合はムーブ コンストラクターが使用されます。