C++為什么要用指針而不直接使用對(duì)象?
問(wèn)題描述
我在使用 C++ 進(jìn)行面向?qū)ο箝_(kāi)發(fā)時(shí),我發(fā)現(xiàn)一個(gè)很讓我非常困惑的問(wèn)題:C++ 中經(jīng)常出現(xiàn)使用對(duì)象指針,而不是直接使用對(duì)象本身的代碼,比如下面這個(gè)例子:
Object *myObject = new Object;
而不是使用:
Object myObject;
要不就是調(diào)用對(duì)象的方法(比如 testFunc())時(shí)不使用這種方式:
myObject.testFunc();
而是得寫(xiě)成這樣:
myObject->testFunc();
我不明白代碼為什么要寫(xiě)成這種形式,我能想到的是指針?lè)绞绞侵苯釉L問(wèn)內(nèi)存,這么寫(xiě)代碼可以提高代碼效率以及執(zhí)行速度,是這樣的么?
最佳回復(fù)來(lái)自 Joseph Mansfield
非常不幸,你在代碼中遇到這么多的動(dòng)態(tài)內(nèi)存分配,但這個(gè)只能說(shuō)明有現(xiàn)在有太多不合格的 C++ 程序員。
這么說(shuō)吧,你的兩個(gè)問(wèn)題本質(zhì)上是同個(gè)問(wèn)題。第一個(gè)問(wèn)題是,應(yīng)該何時(shí)使用動(dòng)態(tài)分配(使用 new 方法)?第二問(wèn)題是,什么時(shí)候該使用指針?
最先要牢記的重點(diǎn)是,你應(yīng)該根據(jù)實(shí)際需求選擇合適的方法。 一般來(lái)說(shuō),使用定義對(duì)象的方式比起使用手工動(dòng)態(tài)分配(或new指針)的方式會(huì)更加合理以及安全。
動(dòng)態(tài)分配
你的提問(wèn)中,所列出的兩種分配對(duì)象方式的主要區(qū)別在于對(duì)象的生存期。通過(guò) Object myObject 方式定義對(duì)象,對(duì)象的生存期是在其作用域內(nèi)自維護(hù)(automatic storage),這個(gè)意味著程序離開(kāi)對(duì)象的作用域之后,對(duì)象將被自動(dòng)銷(xiāo)毀。當(dāng)通過(guò) new Object() 方式分配對(duì)象時(shí),對(duì)象的生存期是動(dòng)態(tài)的,這個(gè)意味著若不顯式地 detete 對(duì)象,對(duì)象將一直存在。你應(yīng)該只在必要的時(shí)候使用動(dòng)態(tài)分配對(duì)象。換句話說(shuō),只要有可能,你應(yīng)該首選定義可自維護(hù)的對(duì)象。
這里是兩個(gè)常見(jiàn)需要?jiǎng)討B(tài)分配對(duì)象的情況:
分配不限制作用域的對(duì)象,對(duì)象存儲(chǔ)在其特定的內(nèi)存中,而不是在內(nèi)存中存儲(chǔ)對(duì)象的拷貝。如果對(duì)象是可以拷貝/移動(dòng)的,一般情況下你應(yīng)該選擇使用定義對(duì)象的方式。
定義的對(duì)象會(huì)消耗大量?jī)?nèi)存,這時(shí)可能會(huì)耗盡??臻g。如果我們永遠(yuǎn)不需要考慮這個(gè)問(wèn)題那該多好(實(shí)際大部分情況下,我們真不需要考慮),因?yàn)檫@個(gè)本身已經(jīng)超出 C++ 語(yǔ)言的范疇,但不幸的是,在我們實(shí)際的開(kāi)發(fā)過(guò)程中卻不得不去處理這個(gè)問(wèn)題。
當(dāng)你確實(shí)需要?jiǎng)討B(tài)分配對(duì)象時(shí),應(yīng)該將對(duì)象封裝在一個(gè)智能指針(smart pointer)或其他提供RAII機(jī)制的類(lèi)型中(類(lèi)似標(biāo)準(zhǔn)的 container)。智能指針提供動(dòng)態(tài)對(duì)象的所有權(quán)語(yǔ)義(ownership),具體可以看一下std::unique_ptr 和 std::shared_ptr 這兩個(gè)例子。如果你使用得當(dāng),基本上可以避免自己管理內(nèi)存(具參見(jiàn) Rule of Zero)。
指針
當(dāng)然,不使用動(dòng)態(tài)分配而采取原始指針(raw pointer)的用法也很常見(jiàn),但是大多數(shù)情況下動(dòng)態(tài)分配可以取代指針,因此一般情況應(yīng)該首選動(dòng)態(tài)分配的方法,除非你遇到不得不用指針的情況。
1. 使用引用語(yǔ)義(reference semantics)的情況。有時(shí)你可能需要通過(guò)傳遞對(duì)象的指針(不管對(duì)象是如何分配的)以便你可以在函數(shù)中去訪問(wèn)/修改這個(gè)對(duì)象的數(shù)據(jù)(而不是它的一份拷貝),但是在大多數(shù)情況下,你應(yīng)該優(yōu)先考慮使用引用方式,而不是指針,因?yàn)橐镁褪潜辉O(shè)計(jì)出來(lái)實(shí)現(xiàn)這個(gè)需求的。注意,采用這種方式,對(duì)象生存期依舊在其作用域內(nèi)自維護(hù)。當(dāng)然,如果通過(guò)傳遞對(duì)象拷貝可以滿足要求的情況下是不需要使用引用語(yǔ)義。
2. 使用多態(tài)的情況。通過(guò)傳遞對(duì)象的指針或引用調(diào)用多態(tài)函數(shù)(根據(jù)入?yún)㈩?lèi)型不同,會(huì)調(diào)用不同處理函數(shù))。如果你的設(shè)計(jì)就是可以傳遞指針或傳遞引用,顯然,應(yīng)該優(yōu)先考慮使用傳遞引用的方式。
3. 對(duì)于入?yún)?duì)象可選的情況,常見(jiàn)的通過(guò)傳遞空指針表示忽略入?yún)?。如果只有一個(gè)參數(shù)的情況,應(yīng)該優(yōu)先考慮使用缺省參數(shù)或是對(duì)函數(shù)進(jìn)行重載。要不然,你應(yīng)該優(yōu)先考慮使用一種可封裝此行為的類(lèi)型,比如 boost::optional或者std::optional
4. 通過(guò)解耦編譯類(lèi)型依賴(lài)減少編譯時(shí)間的情況。使用指針的一個(gè)好處在于可以用于前向聲名(forward declaration)指向特定類(lèi)型(如果使用對(duì)象類(lèi)型,則需要定義對(duì)象),這種方式可以減少參與編譯的文件,從而顯著地提高編譯效率,具體可以看 Pimpl idiom 用法。
5. 與C庫(kù)或C風(fēng)格的庫(kù)交互的情況。此時(shí)只能夠使用指針,這種情況下,你要確保的是指針使用只限定在必要的代碼段中。指針可以通過(guò)智能指針的轉(zhuǎn)換得到,比如使用智能指針的get成員函數(shù)。如果C庫(kù)操作分配的內(nèi)存需要你在代碼中維護(hù)并顯式地釋放時(shí),可以將指針?lè)庋b在智能指針中,通過(guò)實(shí)現(xiàn) deleter 從而可以有效的地釋放對(duì)象。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
- C++中指針函數(shù)與函數(shù)指針的使用
- C++函數(shù)指針和回調(diào)函數(shù)使用解析
- C++實(shí)踐數(shù)組類(lèi)運(yùn)算的實(shí)現(xiàn)參考
- C++實(shí)踐Time類(lèi)中的運(yùn)算符重載參考方法
- C++實(shí)踐分?jǐn)?shù)類(lèi)中運(yùn)算符重載的方法參考
- C++實(shí)踐排序函數(shù)模板項(xiàng)目的參考方法
- C++二維數(shù)組中數(shù)組元素存儲(chǔ)地址的計(jì)算疑問(wèn)講解
- 劍指offer之C++語(yǔ)言實(shí)現(xiàn)鏈表(兩種刪除節(jié)點(diǎn)方式)
- Android Java調(diào)用自己C++類(lèi)庫(kù)的實(shí)例講解
- 一張圖總結(jié)C++中關(guān)于指針的那些事
相關(guān)文章
C++簡(jiǎn)易通訊錄系統(tǒng)實(shí)現(xiàn)流程詳解
這篇文章主要為大家介紹了C語(yǔ)言簡(jiǎn)易版通訊錄的具體實(shí)現(xiàn)流程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05sublime text3搭建配置c語(yǔ)言編譯環(huán)境的詳細(xì)圖解教程(小白級(jí))
這篇文章主要介紹了sublime text3搭建配置c語(yǔ)言編譯環(huán)境,詳細(xì)圖解,小白教程,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01從string類(lèi)的實(shí)現(xiàn)看C++類(lèi)的四大函數(shù)(面試常見(jiàn))
C++類(lèi)一般包括構(gòu)造函數(shù)、拷貝構(gòu)造函數(shù)、析構(gòu)函數(shù)和賦值函數(shù)四大函數(shù),非常常見(jiàn),本文給大家介紹從string類(lèi)的實(shí)現(xiàn)看C++類(lèi)的四大函數(shù),一起看看吧2016-06-06基于C語(yǔ)言實(shí)現(xiàn)計(jì)算生辰八字五行的示例詳解
生辰八字,簡(jiǎn)稱(chēng)八字,是指一個(gè)人出生時(shí)的干支歷日期;年月日時(shí)共四柱干支,每柱兩字,合共八個(gè)字。這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)計(jì)算生辰八字五行的示例代碼,需要的可以參考一下2023-03-03???????C語(yǔ)言實(shí)現(xiàn)單鏈表基本操作方法
這篇文章主要介紹了???????C語(yǔ)言實(shí)現(xiàn)單鏈表基本操作方法,文章圍繞主題展開(kāi)詳細(xì)介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-05-05基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單學(xué)生成績(jī)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單學(xué)生成績(jī)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08QT使用SQLite數(shù)據(jù)庫(kù)超詳細(xì)教程(增刪改查、對(duì)大量數(shù)據(jù)快速存儲(chǔ)和更新)
這篇文章主要給大家介紹了關(guān)于QT使用SQLite數(shù)據(jù)庫(kù)的相關(guān)資料,其中包括增刪改查以及對(duì)大量數(shù)據(jù)快速存儲(chǔ)和更新,SQLite是一種嵌入式關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng),它是一個(gè)軟件庫(kù),提供了一個(gè)自包含、無(wú)服務(wù)器、零配置的、事務(wù)性的SQL數(shù)據(jù)庫(kù)引擎,需要的朋友可以參考下2024-01-01c++優(yōu)先隊(duì)列(priority_queue)用法詳解
這篇文章主要介紹了c++優(yōu)先隊(duì)列(priority_queue)用法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12