iphone開發(fā)內存管理
來源:易賢網 閱讀:897 次 日期:2015-02-02 11:52:29
溫馨提示:易賢網小編為您整理了“iphone開發(fā)內存管理”,方便廣大網友查閱!

開發(fā)iphone 應用程序并不難,基本上就是三個詞 - “memory, memory, memory” 。iphone os 對內存的要求很嚴格,有memory leak ,殺掉; 內存使用超限額,殺掉。一個經過測試的程序,在使用過程中90%以上的崩潰都是內存問題造成的。在這里簡單總結一下object-c 內存管理。

基本概念

object-c 的內存管理基于引用計數(shù)(reference count)這種非常常用的技術。簡單講,如果要使用一個對象,并希望確保在使用期間對象不被釋放,需要通過函數(shù)調用來取得“所有權”,使用結束后再調用函數(shù)釋放“所有權”。“所有權”的獲得和釋放,對應引用計數(shù)的增加和減少,為正數(shù)時代表對象還有引用,為零時代表可以釋放。

函數(shù)

獲得所有權的函數(shù)包括

alloc - 創(chuàng)建對象是調用alloc,為對象分配內存,對象引用計數(shù)加一。

copy - 拷貝一個對象,返回新對象,引用計數(shù)加一。

retain - 引用計數(shù)加一,獲得對象的所有權。

另外,名字中帶有alloc, copy, retain 字串的函數(shù)也都認為會為引用計數(shù)加一。

釋放所有權的函數(shù)包括

release - 引用計數(shù)減一,釋放所有權。如果引用計數(shù)減到零,對象會被釋放。

autorelease - 在未來某個時機釋放。下面具體解釋。

autorelease

在某些情況下,并不想取得所有權,又不希望對象被釋放。例如在一個函數(shù)中生成了一個新對象并返回,函數(shù)本身并不希望取得所有權,因為取得后再沒有機會釋放(除非創(chuàng)造出新的調用規(guī)則,而調用規(guī)則是一切混亂的開始),又不可能在函數(shù)內釋放,可以借助autorelease 。所謂autorelease , 可以理解為把所有權交給一個外在的系統(tǒng)(這個系統(tǒng)實際上叫autorelease pool),由它來管理該對象的釋放。通常認為交給 autorelease 的對象在當前event loop 中都是有效的。也可以自己創(chuàng)建nsautoreleasepool 來控制autorelease的過程。

據蘋果的人說,autorelease效率不高,所以能自己release的地方,盡量自己release,不要隨便交給autorelease來處理。

規(guī)則

引用計數(shù)系統(tǒng)有自己的引用規(guī)則,遵守規(guī)則就可以少出錯:

獲得所有權的函數(shù)要和釋放所有權的函數(shù)一一對應。

保證只有帶alloc, copy, retain 字串的函數(shù)才會讓調用者獲得所有權,也就是引用計數(shù)加一。

在對象的 dealloc函數(shù)中釋放對象所擁有的實例變量。

永遠不要直接調用dealloc來釋放對象,完全依賴引用計數(shù)來完成對象的釋放。

有很多類都提供“便利構造函數(shù)(convenience constructors)”,它們創(chuàng)建對象但并不增加引用計數(shù),意味著不需要調用release來釋放所有權。很好辨認,它們的名字中不會有alloc和copy。

只要遵守這些規(guī)則,基本上可以消除所有intrument可以發(fā)現(xiàn)的內存泄露問題。

容器

類似nsarray, nsdictionary, nsset 等類,會在對象加入后引用計數(shù)加一獲得所有權,在對象被移除或者整個容器對象被釋放的時候釋放容器內對象的所有權。類似的情況還有uiview對 subview的所有權關系,uinavigationcontroller對其棧上的controller的所有權關系等等。

其他所有權的產生

還有一些用法會讓系統(tǒng)擁有對象的所有權。比如nsobject 的performselector:withobject:afterdelay 。如果有必要,需要顯示的調用cancelpreviousperformrequestswithtarget:selector:object: ,否則有可能產生內存泄露。

因這種原因產生的泄露因為并不違反任何規(guī)則,是intrument所無法發(fā)現(xiàn)的。

循環(huán)引用

所有的引用計數(shù)系統(tǒng),都存在循環(huán)應用的問題。例如下面的引用關系:

對象a創(chuàng)建并引用到了對象b.

對象b創(chuàng)建并引用到了對象c.

對象c創(chuàng)建并引用到了對象b.

這時候b和c的引用計數(shù)分別是2和1。當a不再使用b,調用release釋放對b的所有權,因為c還引用了b,所以b的引用計數(shù)為1,b不會被釋放。b不釋放,c的引用計數(shù)就是1,c也不會被釋放。從此,b和c永遠留在內存中。

這種情況,必須打斷循環(huán)引用,通過其他規(guī)則來維護引用關系。比如,我們常見的delegate往往是assign方式的屬性而不是retain方式的屬性,賦值不會增加引用計數(shù),就是為了防止delegation兩端產生不必要的循環(huán)引用。如果一個uitableviewcontroller 對象a通過retain獲取了uitableview對象b的所有權,這個uitableview對象b的delegate又是a, 如果這個delegate是retain方式的,那基本上就沒有機會釋放這兩個對象了。自己在設計使用delegate模式時,也要注意這點。

因為循環(huán)引用而產生的內存泄露也是instrument無法發(fā)現(xiàn)的,所以要特別小心。

一些和內存管理相關的有用內容:

objective-c2.0 號稱可以支持garbage collection了, 也就是垃圾回收,但是我還沒在xcode以及文檔中找到相關的用法,也懶得去查了。 關于garbage collection的內容也沒啥可說的, 想說說這幾天遇到的無gc情況下的幾個內存相關問題。

objective-c 的autorelease確實給開發(fā)省了不少事情提高了開發(fā)效率, 這對于mac osx桌面開發(fā)沒問題,因為內存大不存在內存緊張的局面。但是如果要為iphone開發(fā)程序, 還是慎用 autorelease的好, 否則只好等程序退出時再清理內存了。 在iphone上最好不要保存不必要的對象, 使用的時候在創(chuàng)建,比如圖片、文件等等。 還有一點要注意的就是uitableview, 如果你是把uitableview放在uitableviewcontroller中,那么別擔心,沒有什么問題,如果你是在 uiviewcontroller或者其子類里放置uitableview,那么注意了, 在pop掉 viewcontroller的時候一定記得先把uitableview的delegate設置為空, 也就是 [tableview setdelegate:nil] ,之所以這樣做, 是因為tableview的delegate是個retain,會保存對象, 所以如果你不在pop之前將delegate設為nil, 將不會調用view controller的dealloc,內存也就無法釋放, 這么來幾下恐怕就要內存吃緊了。

總結了幾條內存使用經驗

1. 對象現(xiàn)用現(xiàn)創(chuàng)建

2. 所有用alloc,new , retain等創(chuàng)建的對象都需要調用release去釋放, 千萬別發(fā)送release消息給autorelease對象, 否則只能over了

3. 注意delegate,如果時retain類型,最好在釋放之前將之設為nil

4. 在頻繁使用alloc的地方(循環(huán)) 創(chuàng)建自己的nsautoreleasepool

5. 對于uiimage對象慎用 [uiimage imagenamed:], 使用[uiimage imagewithcontentoffile:] 或者[image initwithcontentoffile:]

更多信息請查看IT技術專欄

更多信息請查看網絡編程
易賢網手機網站地址:iphone開發(fā)內存管理

2026上岸·考公考編培訓報班

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關于我們 | 聯(lián)系我們 | 人才招聘 | 網站聲明 | 網站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權所有:易賢網