C++簡明分析臨時對象是什么
一、初探臨時對象
1.問題
下面的程序輸出什么?為什么?
下面編寫程序進行實驗:
#include <stdio.h> class Test { int mi; public: Test(int i) { mi = i; } Test() { Test(0); } void print() { printf("mi = %d\n", mi); } }; int main() { Test t; t.print(); return 0; }
輸出結(jié)果如下:
程序意圖:
- 在 Test() 中以 0 作為參數(shù)調(diào)用 Test(int i)
- 將成員變量 mi 的初始值設(shè)置為 0
運行結(jié)果:
- 成員變量 mi 的值為隨機值
2.思考
構(gòu)造函數(shù)是一個特殊的函數(shù)
- 是否可以直接調(diào)用?
- 是否可以在構(gòu)造函數(shù)中調(diào)用構(gòu)造函數(shù)?
- 直接調(diào)用構(gòu)造函數(shù)的行為是什么?
3.答案
- 直接調(diào)用構(gòu)造函數(shù)將產(chǎn)生一個臨時對象
- 臨時對象的生命周期只有一條語句的時間(過了這條 C++ 語句,臨時對象將被析構(gòu)而不復(fù)存在)
- 臨時對象的作用域只在一條語句中
- 臨時對象是 C++ 中值得警惕的灰色地帶
可以將上面代碼寫成這樣,避免臨時對象:
#include <stdio.h> class Test { int mi; void init(int i) { mi = i; } public: Test(int i) { init(i); } Test() { init(0); } void print() { printf("mi = %d\n", mi); } }; int main() { Test t; t.print(); return 0; }
輸出結(jié)果如下:
再來看一個程序,深刻體會一下臨時對象:
#include <stdio.h> class Test { int mi; void init(int i) { mi = i; } public: Test(int i) { printf("Test(int i)\n"); init(i); } Test() { printf("Test()\n"); init(0); } void print() { printf("mi = %d\n", mi); } ~Test() { printf("~Test()\n"); } }; int main() { printf("main begin\n"); Test(); Test(10); printf("main end\n"); return 0; }
輸出結(jié)果如下:
這個程序很好的說明了臨時對象的生命周期只有一條語句的時間(過了這條 C++ 語句,臨時對象將被析構(gòu)而不復(fù)存在)
二、編譯器的行為
現(xiàn)代 C++ 編譯器在不影響最終執(zhí)行結(jié)果的前提下,會盡力減少臨時對象的產(chǎn)生!?。?/p>
下面來看一個例子:
#include <stdio.h> class Test { int mi; public: Test(int i) { printf("Test(int i) : %d\n", i); mi = i; } Test(const Test& t) { printf("Test(const Test& t) : %d\n", t.mi); mi = t.mi; } Test() { printf("Test()\n"); mi = 0; } int print() { printf("mi = %d\n", mi); } ~Test() { printf("~Test()\n"); } }; Test func() { return Test(20); } int main() { //Test t(10); 等價于 Test t = Test(10); Test t = Test(10); // ==> Test t = 10; Test tt = func(); // ==> Test tt = Test(20); ==> Test tt = 20; t.print(); tt.print(); return 0; }
輸出結(jié)果如下:
注意兩點:
- 通過輸出結(jié)果可以看到【編譯器并沒有按照生成臨時對象,再用臨時對象初始化 t 對象(其中涉及調(diào)用拷貝構(gòu)造函數(shù))】的步驟,因為現(xiàn)代的編譯器都會盡力避免臨時對象的產(chǎn)生。
- Test t = Test(10); 等價于 Test t = 10; 寫成Test t = 10; 可以杜絕臨時對象的產(chǎn)生。因為臨時對象的產(chǎn)生會帶來性能上的問題,Test t = Test(10); 相當(dāng)于調(diào)用了兩次構(gòu)造函數(shù), 而 Test t = 10; 少調(diào)用一次函數(shù),性能得到提升。
三、小結(jié)
- 直接調(diào)用構(gòu)造函數(shù)將產(chǎn)生一個臨時對象
- 臨時對象是性能的瓶頸,也是 bug 的來源之一
- 現(xiàn)代 C++ 編譯器會盡力避開臨時對象
- 實際工程開發(fā)中需要人為的避開臨時對象
到此這篇關(guān)于C++簡明分析臨時對象是什么的文章就介紹到這了,更多相關(guān)C++臨時對象內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Qt+GDAL庫實現(xiàn)制作經(jīng)緯度坐標(biāo)轉(zhuǎn)換工具
這篇文章主要為大家詳細(xì)介紹了如何利用Qt和GDAL庫實現(xiàn)制作經(jīng)緯度坐標(biāo)轉(zhuǎn)換工具,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下2023-04-04C 程序?qū)崿F(xiàn)密碼隱秘輸入的實例 linux系統(tǒng)可執(zhí)行
下面小編就為大家?guī)硪黄狢 程序?qū)崿F(xiàn)密碼隱秘輸入的實例 linux系統(tǒng)可執(zhí)行。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11