プログラムの最適化
#include <iostream> using namespace std; class CTempValue { public: int val1; int val2; public: CTempValue(int v1 = 0, int v2 = 0) :val1(v1), val2(v2) //构造函数 { cout << "调用了构造函数!" << endl; cout << "val1 = " << val1 << endl; cout << "val2 = " << val2 << endl; } CTempValue(const CTempValue& t) :val1(t.val1), val2(t.val2) //拷贝构造函数 { cout << "调用了拷贝构造函数!" << endl; } virtual ~CTempValue() { cout << "调用了析构函数!" << endl; } }; CTempValue Double(CTempValue& ts) { CTempValue tmpm; tmpm.val1 = ts.val1 * 2; tmpm.val2 = ts.val2 * 2; return tmpm; } int main() { CTempValue ts1(10, 20); Double(ts1); return 1; }
- 外部から誰も返さない一時オブジェクト
- 出力結果:
- 外部から誰かによって返された一時的なオブジェクト
- 一時オブジェクトはts2に組み込まれています
int main() { CTempValue ts1(10, 20); CTempValue ts2 = Double(ts1); return 1; }
- 出力結果:
- 分析コード
CTempValue Double(CTempValue &ts) { CTempValue tmpm; //消耗一个构造函数,一个析构函数 tmpm.val1 = ts.val1 * 2; tmpm.val2 = ts.val2 * 2; return tmpm; //生成一个临时对象,然后调用拷贝构造函数把tmpm的内容拷贝构造到这个临时对象中去,然后返回临时对象。 //这个临时对象也消耗了一个拷贝构造函数 ,(如果外部没人接)消耗了一个析构函数; }
- 最適化
CTempValue Double(CTempValue &ts) { return CTempValue(ts.val1 * 2, ts.val2 * 2); //生成一个临时对象。 }
- 出力結果:
- 1つの不足している構造とデストラクタ
開発者とコンパイラーの視点からの分析
- 開発者の視点:
CTempValue Double(CTempValue &ts) { return CTempValue(ts.val1 * 2, ts.val2 * 2); //生成一个临时对象。 } int main(){ CTempValue ts1(10, 20); CTempValue ts2 = Double(ts1); Double(ts1); return 1; }
- コンパイラの視点:
void Double(CTempValue &tmpobj, CTempValue &ts) //编译器会插入第一个参数 { tmpobj.CTempValue::CTempValue(ts.val1 * 2, ts.val2 * 2); return; } int main(){ CTempValue ts1; //分配存储空间,不调用构造函数 ts1.CTempValue::CTempValue(10, 20); CTempValue tmpobj; Double(tmpobj, ts1); return 1; }
コンパイラーレベルからの最適化
- Linuxコンパイラg ++は、一時オブジェクトを返す状況に合わせて最適化されています。NRV最適化(名前付き戻り値)。RVO(戻り値の最適化);
- 最適化をオフにします。
g++ -fno-elide-constructors xxx.cpp -o xxx
- Visual Studioに最適化された設定
- 最適化の手順:
- (1)コンパイラが本当に最適化されているかどうかはわかりませんが、知るためにはさまざまなテストを行う必要があります
- (2)コードが非常に複雑な場合、コンパイラーは最適化しないことをあきらめることがあります。
- (3)最適化しすぎないでください。
- (4)最適化によりミスが発生する可能性があります