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