複製省略
複製省略或者譯作「省略不必要的複製」(copy elision),是C++語言標準中定義的編譯優化技術。
當一個class類型的temporary object的臨時對象用於初始化同類型的對象[1]時,複製初始化通常優化為直接初始化;但在語義上仍然需要複製構造函數是可訪問的。[2] 例如:
#include <iostream>
struct C {
explicit C(int) {}
C(const C&) { std::cout<<"copy constructor"<<std::endl; } // the copy constructor has a visible side effect
}; // it modifies an object with static storage duration
int main() {
C c1(42); // direct-initialization, calls C::C(42)
C c2 = C(42); // copy-initialization elision as direct-initialization, so as to call C::C(42)
return 0;
}
上例,在g++與Visual C++默認都是複製省略。
throw語句拋出一個異常對象,catch語句匹配一個異常對象,默認是要執行複製構造函數的。如下例:
#include <iostream>
struct C {
C() {}
C(const C&) { std::cout << "Hello World!\n"; }
};
void f() {
C c;
throw c; // copying the named object c into the exception object.
} // 由于异常对象保存在线程信息块TIB中,这里的复制构造不可省略. 这使得调用栈上压栈保存的函数都有机会处理这个异常对象
int main() {
try {
f();
}
catch(C c) { // copying the exception object into the temporary in the exception declaration.
} // 由于这里是传值的catch批配,因此复制初始化不可省略. 通常用于catch块可能会修改异常对象的内容,但又需要把原来的异常对象重新抛出(re-throw)
}
上例默認應該列印輸出兩次"Hello World!"
GCC的編譯選項-fno-elide-constructors關閉複製省略。
參考文獻
- ^ ISO/IEC (2003). ISO/IEC 14882:2003(E): Programming Languages - C++ §12.8 Copying class objects [class.copy] para. 15
- ^ Sutter, Herb. More Exceptional C++. Addison-Wesley. 2001.