[原創文章歡迎轉載,但請保留作者信息]
Justin 于 2010-05-28
其實平時很多時候都會看到重載new和delete的情況,不過應該沒有很全面的歸納過為什么要這么做。于是大師來做了這件事:
- 重載new和delete的一個原因是可以在分配的內存塊前加上特定的信息(如內存ID號),到釋放的時候delete檢查將要釋放的內存是否符合釋放的條件(如是否已經釋放過了一次;是否大小與申請時不吻合;是否有權釋放,等等)。曾經工作過的公司就有這樣重載new和delete的方法,組里某牛還曾在每塊內存前面加上一段分配內存時的堆棧內容,于是解決了許多豪無頭緒的內存泄漏問題。
- 對這對操作符的重載通常可以提高效率。默認的new和delete的目標是通用,它們需要考慮到所有可能的內存分配/釋放情況,就需要為這些林林總總的情況實現應對的代碼。于是
- 額外的代碼用于保證多線程安全,
- 額外的空間用于跟蹤內存分配,
- 對是否需要地址對齊的不確定導致可能的效率下降,
- 由于目標系統的不明確,一些可以綁定在一起分配的數據沒辦法做到,
- 由于目標系統的不明確,一些特定的需求(比如說將釋放的內存塊全部寫0)無法實現,
于是就變成一個任何人都可以用,但沒有一個人會認為是最好用的new+delete。重載new和delete,可以剔除那些對自己無用的處理機制,保留甚至加強自己需要的功能,于是就提高了內存分配/釋放的效率。當然,可以這樣做的前提就是對自己的系統很了解。
- 用重載的new和delete來收集系統內存使用數據。在真正設計內存相關行為之前,可以借由重載的new和delete來收集各類數據,因為它們是目擊每次內存分配和釋放的證人。
不過,雖然重載new和delete有以上諸多好處,大師還是建議:如非必需,還是不要自己寫new/delete這樣的內存管理程序。因為要寫這樣的程序不難,要寫好就不容易。
上課的時候大師舉了個比較常見的例子,就是在重載new時,如果沒有考慮到內存地址對齊(alignment)的問題,寫出來的程序就有可能崩潰,或者,至少會跑得比較慢。
所以,在想要重載new和delete的時候,先看看是不是已經有下面的更簡單安全有效的選項:
- 編譯器有開關可以提供需要的功能,如調試/內存日志等,沒有必要自己做
- 很多商用平臺已經提供了成熟的new和delete可以重載,只要你買了(或是你的公司買了),就可以直接來用。
- 再有就是有很多開源的內存管理工具可以借用。大師推薦Boost的Pool Library就是其中之一。