c++ 性能優化策略


c++ 性能優化策略

作者:D_Guco 
來源:CSDN 
原文:https://blog.csdn.net/D_Guco/article/details/75729259 

 

1 關於繼承:不可否認良好的抽象設計可以讓程序更清晰,代碼更看起來更好,但是她也是有損失的,在繼承體系中子類的創建會調用父類的構造函數,銷毀時會調用父類的析構函數,這種消耗會隨着繼承的深度直線上升,所以不要過度的抽象和繼承,更為嚴重的是當多重繼承中並且有虛函數的存在時情況更為復雜,的確,這些問題涉及開銷,但是,多重繼承減少了編碼的負擔,同時也讓問題的解決方案更加簡潔,這當然要付出一些代價.總之,與n個基類的多重繼承層次相關的額外虛函數表有n-1個。派生類和最左邊的非虛基類共享同一個虛函數表。因此,帶有2個基類的多重繼承層次,有1個(2-1=1)基類的虛函數表和1個派生類的虛函數表(最左邊的基類與派生類共享該虛函數表),總共有2個虛函數表,如果有虛繼承的存在,會進一步增長這個過程,它是有額外的開銷的。

2 對象的復合:對象的復合和繼承很相似,當一個對象包含其他對象構造時也會引起額外的構造。關於這點可能會有很多人不解,認為這是不可避免的,舉個例子,比如你的類A中包含了類B非指針和引用對象,那么在你構造對象a的時候會自動調用b的無參構造函數,即使你還沒有用到她,用指針代替就沒有這種消耗,另外如果你的一個對象中用到數組和字符串,你是選擇string和vector還是char* 和c系的數組呢,如果沒有用到c++stl庫提供的相關的高級用法,建議選擇后者。

3 構造函數:盡量用參數列表初始化代替參數,避免值傳遞初始化。

4 變量延時定義:從c系轉過來的仍保留着c的習慣,在函數第一行先把所有用到的變量都定義好,但是c是沒有運行時的消耗的,對於c++時不一樣的,對於c++對象的構造和銷毀時有消耗的,如果有大量的對象只在某個if條件的一個分支中出現,那就會有50%的情況這些消耗是可以避免的。對於這點在一個類中也是一樣的,如果成員中有成員只在某個時刻能用,就用指針代替,在構造對象時初始化成空指針,避免構造時調用他的構造函數。

5 虛函數:虛函數的底層實現是通過一個虛函數表來實現的,因此有虛函數的類構造時必須先初始化虛函數表,函數調用時也必須先找到虛函數表,然后通過指針偏移找到相應的函數,通常情況下調用虛函數是沒有運行時消耗的,但是根據編譯器的實現不同,在調用虛函數時,有些調用可能導致增加虛函數表大小的額外開銷,或者只有那些需要調整this指針的調用才會發生額外的運行開銷,但不會增加虛函數表的大小,在多重繼承和虛基類的時候這種消耗會顯著增加,關於繼承已經提過,所以避免濫用虛函數和虛繼承,有時候可以用模版設計來代替虛繼承,把運行時的消耗提前到編譯期。關於虛函數的消耗:點擊打開鏈接

6 返回值優化: 雖然c++編譯器會選擇性的進行RVO優化但是不是強制的,當函數有多個返回語句並且返回不通名稱的對象,函數過於復雜,返回對象沒有定義拷貝構造函數時,rvo優化是不會執行的,所以當函數返回一個很大的對象時在不確定rvo優化會執行時,盡量避免值傳遞。

7 變量的定義:在定義變量時盡量避免類型的不匹配造成臨時變量的產生。

8 內存管理:c++內存管理的大權由我們自己掌握,對於項目中要頻繁申請和釋放的對象建議用簡單的內存池來管理,可以大大的降低頻繁申請和釋放內存帶來的消耗。

9 善用內聯:內聯函數不僅僅是簡單的函數調用似的優化,他還有一個最大的優點就是,可以讓編譯期進行進行邊界代碼的運行環境優化,內聯把代碼拷貝到執行環境處避免了函數調用帶來的消耗,並且編譯期可以進行正常的編譯優化,而函數調用是不能實現的。

10 stl :記住一點stl不是唯一的選擇,有時候也不是最好的選擇,合理選擇stl善用stl算法。

11 緩存:對於多次使用的計算結果及時緩存,避免重復計算。

12 延時計算:對於不關心計算結果的計算過程盡量延時執行或者異步去執行。

13 多線程:盡可能的使用無鎖式多線程開發,鎖是一個非常消耗性能的東西,保證數據同步的手段有很多,voalite,原子操作都可已實現,盡量通過一些技巧使用這些手段避免所得使用,如果迫不得已要使用鎖,盡量減少鎖的消耗,比如降低鎖的粒度,使用性能更高的鎖等等。

14 std::move操作: 當不得不進行深拷貝時,如果深拷貝數據源在拷貝后就不在使用,盡可能的用move操作代替,或者在參數傳遞時用move操作代替臨時的實參變量。

15 cpu緩存:合理的利用cpu cache可以極大的提高代碼的運行效率(例如:數組中以每列遍歷和每行遍歷的效率的不同),當然多線程環境下也要考慮cpu cache帶來的影響。

16 內存對齊:在進行網絡編程時,最好對網絡中傳送的數據快進行內存補齊,通常是8字節對其,提高cpu訪問內存效率,從而提高數據讀寫速度。

17 函數參數:用const引用代替值傳遞,如果函數參數過多,可以用對象來打包參數,減少參數過多帶來的性能消耗。

18 算法: 盡可能的優化你的算法。

19 關於智能指針:對於智能指針我的選擇是必須用,它可以大大降低程序的crash頻率,但是智能指針的和普通指針相比是有額外的消耗的,她的底層是一個原子操作來來統計引用數和一個普通指針,雖然原子操作和鎖相比性能高了不少但是和普通的加減操作還是慢了不少,智能指針的大小為16個字節,而普通指針的大小只有4個字節,拷貝的成本也不一樣,所以在使用正確的情況下可以使用智能指針的引用來減少拷貝的消耗(注意這里的前提是正確的使用引用,不要引用以一個即將被銷毀的變量)。

20 內存池:對於需要頻繁申請和釋放的內存對象,如果可以重復利用對象的內存,強烈建議通過內存池或者重載對象的new操作符或者重載對象的placement new操作符來減少頻繁的申請和釋放內存,從而減少申請和釋放內存的消耗和內存碎片的產生。

21 其他優化方案:位運算代替乘除法,前綴運算符代替后綴運算等等。

=============== End

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM