iOS內(nèi)存管理-基本概念整理

主要內(nèi)容:

  1. 內(nèi)存區(qū)域劃分
  2. 內(nèi)存管理/引用計(jì)數(shù)
  3. MRC手動(dòng)管理引用計(jì)數(shù)
  4. ARC自動(dòng)引用計(jì)數(shù)
  5. 內(nèi)存泄漏問題
  6. 野指針問題

一、內(nèi)存區(qū)域劃分

程序在分配內(nèi)存時(shí),主要分為:棧區(qū)、堆區(qū)、靜態(tài)區(qū)、常量區(qū)、代碼區(qū);

內(nèi)存區(qū)域 具體說明
棧區(qū) 存放局部變量的值,系統(tǒng)自動(dòng)分配和釋放;
特點(diǎn):容量小,速度快,有序
堆區(qū) 存放通過malloc系列函數(shù)或new操作符分配的內(nèi)存,如對(duì)象;
一般由程序員分配和釋放,如果不釋放,則出現(xiàn)內(nèi)存泄露;
特點(diǎn):容量大,速度慢,無(wú)序;
靜態(tài)區(qū) 存放全局變量和靜態(tài)變量(包括靜態(tài)局部變量和靜態(tài)全局變量);
當(dāng)程序結(jié)束時(shí),系統(tǒng)回收;
常量區(qū) 存放常量的內(nèi)存區(qū)域;
程序結(jié)束時(shí),系統(tǒng)回收;
代碼區(qū) 存放二進(jìn)制代碼的區(qū)域

從上述分類上看,我們?cè)陂_發(fā)過程中主要涉及的是堆上內(nèi)存的管理。通常,我們創(chuàng)建一個(gè)對(duì)象的代碼如下:

NSObject *obj = [[NSObject alloc] init]; 

上述代碼創(chuàng)建了一個(gè)NSObject類型的指針obj和一個(gè)NSObject類型的對(duì)象。obj指針存在棧上,而其指向的對(duì)象則是在堆上。這種對(duì)象也稱之為堆對(duì)象

二、內(nèi)存管理/引用計(jì)數(shù)

無(wú)論是MRC還是ARC環(huán)境,Objective-C都采用引用計(jì)數(shù)來管理內(nèi)存;每個(gè)對(duì)象都有一個(gè)引用計(jì)數(shù)器,任何時(shí)候指向?qū)ο蟮闹羔槀€(gè)數(shù)和對(duì)象的引用計(jì)數(shù)相等,當(dāng)一個(gè)對(duì)象的引用計(jì)數(shù)為0的時(shí)候?qū)?huì)被釋放;

OC管理內(nèi)存涉及到對(duì)象的"生成"、"持有"、"釋放",MRC需要調(diào)用對(duì)應(yīng)的方法來管理引用計(jì)數(shù),而ARC則是自動(dòng)管理引用計(jì)數(shù),無(wú)需再調(diào)用這些內(nèi)存管理的方法。雖然兩者管理內(nèi)存的形式不同,但是它們都遵循相同的內(nèi)存管理規(guī)律,內(nèi)容如下:

  1. 自己生成的對(duì)象,自己所持有;
  2. 非自己生成的對(duì)象,自己也能持有;
  3. 不再需要自己持有對(duì)象時(shí)釋放;
  4. 非自己持有的對(duì)象無(wú)法釋放;

三、MRC手動(dòng)管理引用計(jì)數(shù)

MRC,即手動(dòng)管理引用計(jì)數(shù)。當(dāng)我們通過allocretain等方法持有對(duì)象后,也必須有相應(yīng)的release或者autorelease將其釋放。總結(jié)對(duì)象操作與Objective-C內(nèi)存方法對(duì)應(yīng)關(guān)系如下:

對(duì)象操作 OC方法
生成并持有對(duì)象 alloc/new/copy/mutableCopy等名稱開頭方法
持有對(duì)象 retain方法
釋放對(duì)象 release方法
廢棄對(duì)象 dealloc方法
1.自己生成的對(duì)象,自己所持有/非自己生成對(duì)象,不持有
id obj = [[NSObject alloc] init];  //自己生成并持有對(duì)象
id obj1 = [NSMutableArray array];  //取得非自己生成的對(duì)象,但不持有對(duì)象

OC中使用allocnewcopymutableCopy這些名稱開頭的方法意味著自己生成對(duì)象并持有,否則就是非自己生成的對(duì)象不持有。如上源碼,使用NSObject類的alloc類方法就能自己生成并持有對(duì)象,指向生成并持有對(duì)象的指針被賦值給了obj

通過自定義方法來理解這兩種創(chuàng)建對(duì)象方法的區(qū)別(系統(tǒng)方法也是類似的實(shí)現(xiàn)),測(cè)試代碼如下:

//以alloc開頭的方法
- (id)allocObject {
    id obj = [[NSObject alloc] init];
    return obj;
}

- (id)object {
    id obj = [[NSObject alloc] init];
    [obj autorelease];  //用該方法,可以使取得的對(duì)象存在,但是自己不持有對(duì)象
    return obj;
}

autorelease即自動(dòng)釋放,對(duì)象已經(jīng)加入自動(dòng)釋放池,所以獲取對(duì)象并不持有;涉及到的自動(dòng)釋放池的內(nèi)容會(huì)在后續(xù)詳細(xì)總結(jié)。

注意:生成并持有對(duì)象的的方法一定是駝峰拼寫來命名的方法,如allocallocMyObject等方法;相反allocatemutableCopyed就不屬于這類方法;

2.非自己生成的對(duì)象,自己也能持有
 id obj1 = [NSMutableArray array];  //取得非自己生成的對(duì)象,但不持有對(duì)象
 [obj retain];                      //通過retain方法,持有了對(duì)象

源代碼中,NSMutableArray類對(duì)象被賦值給變量obj,但是變量obj自己不持有該對(duì)象。使用retain方法后可以持有對(duì)象。

3.不再需要自己持有對(duì)象時(shí)釋放

自己持有的對(duì)象,一旦不需要,持有者有義務(wù)釋放該對(duì)象,釋放對(duì)象使用release方法。

id obj = [[NSObject alloc] init]; //自己生成并持有對(duì)象
[obj release];                    //釋放自己持有的對(duì)象
NSLog(@"%@",obj);                 //已經(jīng)釋放,再次使用會(huì)崩潰

雖然指向?qū)ο蟮闹羔樢廊槐A粼谧兞?code>obj中,看似可以訪問,但對(duì)象一經(jīng)釋放就絕不可再訪問。

4.非自己持有的對(duì)象無(wú)法釋放

在應(yīng)用程序中釋放非自己持有的對(duì)象就會(huì)造成崩潰,使用代碼演示如下:

//情況1:釋放完不再需要的對(duì)象后再次釋放,訪問了已經(jīng)廢棄的對(duì)象而崩潰!
id obj = [[NSObject alloc] init];
[obj release];
[obj release];

//情況2:取得自己并不持有的對(duì)象對(duì)其釋放,釋放了非自己持有的對(duì)象而崩潰!
id obj = [[NSMutableArray array];
[obj release];

四、ARC自動(dòng)引用計(jì)數(shù)

ARC(Automic Reference Counting),即自動(dòng)引用計(jì)數(shù);這是iOS5推出的新特性,iOS4.3也支持ARC,只是不能使用weakARC不再需要使用類似retainrelease的操作來持有或者釋放對(duì)象,從而大大提高了開發(fā)效率;

1.ARC使用條件
  1. Xcode4.2或以上版本
  2. 使用LLVM編輯器3.0或以上版本
  3. Xcode編譯器選項(xiàng)中設(shè)置ARC有效
2.ARC基本原理
  1. ARC下的編譯器會(huì)在代碼編譯階段合適的位置,自動(dòng)加入retain/release/autorelease的操作;
  2. ARC的規(guī)則:只要還有一個(gè)強(qiáng)引用指針指向?qū)ο螅瑢?duì)象就會(huì)保存在內(nèi)存中;
  3. ARC中使用strongweak關(guān)鍵字來修飾對(duì)象;strong表示強(qiáng)引用,對(duì)應(yīng)MRC下的retainweak表示弱引用,對(duì)應(yīng)原來的assign,不同的是當(dāng)對(duì)象被釋放的時(shí)候,對(duì)象weak指針自動(dòng)賦值為nil,從而不會(huì)引發(fā)野指針錯(cuò)誤;
3.ARC所有權(quán)修飾符

ARC有效時(shí),OC處理id類型和對(duì)象類型必須附加所有權(quán)修飾符。所有權(quán)修飾符一共有四種:

  • __strong
  • __weak
  • __unsafe_unretained
  • __autoreleasing

__strong修飾符:
__strongid類型和對(duì)象類型默認(rèn)的所有權(quán)修飾,表示對(duì)對(duì)象的"強(qiáng)引用";當(dāng)對(duì)象沒有任何一個(gè)強(qiáng)引用指向它的時(shí)候,對(duì)象將被釋放。

__weak修飾符:
1.__weak__strong修飾符的作用相反,表示弱引用,不會(huì)增加引用計(jì)數(shù);
2.當(dāng)對(duì)象被釋放后,所有指向它的弱引用都會(huì)被置為nil,這樣避免了野指針問題。
3.__weak修飾符常用于解決循環(huán)引用問題;
4.__weak只能用于iOS5以上版本,更早的版本只能使用__unsafe_unretained修飾符。

__unsafe_unretained修飾符:
1.__unsafe_unretained提供弱引用,與__weak作用類似;
2.__unsafe_unretained不能在對(duì)象釋放后自動(dòng)置為nil,易產(chǎn)生野指針問題;
3.__unsafe_unretained可用于iOS5之前版本,為兼容ARC弱引用而引入;

__autoreleasing修飾符:
將對(duì)象賦值給附有__autoreleasing修飾符的變量,
等同于在MRC下調(diào)用對(duì)象的autorelease方法,即對(duì)象被注冊(cè)到autoreleasepool

ARC環(huán)境不能使用NSAutoreleasePool類也不能調(diào)用autorelease方法,代替它們實(shí)現(xiàn)對(duì)象自動(dòng)釋放的是@autoreleasepool塊__autoreleasing修飾符;兩種環(huán)境下的使用情況類比如下圖:

對(duì)比MRC與ARC的自動(dòng)釋放池使用.png

如圖所示,@autoreleasepool塊替換了NSAutoreleasePoool類對(duì)象的生成、持有及廢棄這一過程。而附有__autoreleasing修飾符的變量替代了autorelease方法,將對(duì)象注冊(cè)到了autoreleasepool;

但事實(shí)上,顯式使用__autoreleasing修飾符的情況非常少見,這主要是因?yàn)?code>ARC的很多情況下,即使是不顯式的使用__autoreleasing,也能實(shí)現(xiàn)對(duì)象被注冊(cè)到釋放池中。這其中就包括以下的幾種情況:

  1. 編譯器檢查方法名是否以alloc/new/copy/mutableCopy開始,如果不是則自動(dòng)將返回對(duì)象注冊(cè)到autoreleasepool;
  2. 訪問附有__weak修飾符的變量時(shí),實(shí)際上必定要訪問注冊(cè)到autoreleasepool的對(duì)象;
  3. id的指針或?qū)ο蟮闹羔樤跊]有顯式地指定修飾符時(shí)候,會(huì)被默認(rèn)附加上__autoreleasing修飾符;
4.ARC屬性修飾符

ARC中的所有權(quán)修飾與屬性修飾符存在著對(duì)應(yīng)關(guān)系,如果不一致還會(huì)引起編譯錯(cuò)誤。總結(jié)兩者的對(duì)應(yīng)關(guān)系如下:

屬性修飾符 所有權(quán)修飾符
assign __unsafe_unretained
copy __strong(但是賦值的是被復(fù)制的對(duì)象)
retain __strong
strong __strong
unsafe_unretained __unsafe_unretained
weak __weak

以上各種屬性只有copy不是簡(jiǎn)單的賦值,它賦值的是通過NSCopying接口的copyWithZone:方法復(fù)制賦值源生成的對(duì)象。

5.ARC管理內(nèi)存的規(guī)則
  1. 不能使用retain/release/retainCount/autorelease內(nèi)存管理方法;
  2. 不能使用NSAllocateObject/NSDeallocateObject方法;
  3. 必須遵守內(nèi)存管理的方法命名規(guī)則;
  4. 不能顯式調(diào)用dealloc方法,如[super dealloc];
  5. 使用@autoreleasepool塊代替NSAutoreleasePool;
  6. 不能使用區(qū)域(NSZone);
  7. 對(duì)象類型變量不能作為C語(yǔ)言結(jié)構(gòu)體(struct/union)的成員;
  8. 顯式轉(zhuǎn)換idvoid *
6.必須遵守內(nèi)存管理的方法命名規(guī)則

MRC 下,用于對(duì)象生成/持有的方法必須遵守allocnewcopymutableCopy的命名規(guī)則。以這些名稱開始的方法在返回對(duì)象時(shí),必須返回給調(diào)用方所應(yīng)當(dāng)持有的對(duì)象。這在ARC環(huán)境下的規(guī)則一樣。只是ARC下關(guān)于init開發(fā)的方法規(guī)則要更加嚴(yán)格了:

  1. 必須是實(shí)例方法,且返回對(duì)象;
  2. 返回對(duì)象應(yīng)該是id類型或該方法聲明類的對(duì)象,抑或該類的超類或子類;
  3. 該返回類型不注冊(cè)到autoreleasepool上;
  4. 基本上,init方法只是對(duì)alloc方法返回值的對(duì)象進(jìn)行初始化處理并返回對(duì)象;
7.顯式轉(zhuǎn)換id和void *

這里說到的其實(shí)就是Core FoundationFoundation兩者之間的轉(zhuǎn)換。
Core Foundation是由C語(yǔ)言實(shí)現(xiàn)的,而FoundationObjective-C實(shí)現(xiàn),兩者可以相互轉(zhuǎn)換。

MRC不存在顯式轉(zhuǎn)換的問題,因?yàn)楸緛砭褪鞘謩?dòng)管理內(nèi)存。但是為了在ARC也能實(shí)現(xiàn)對(duì)Core Foundation對(duì)象的自動(dòng)內(nèi)存管理,我們就必須將其與Objective-C對(duì)象的轉(zhuǎn)換。Objective-C中提供了三個(gè)關(guān)鍵字__bridge__bridge_retained__bridge_transfer來實(shí)現(xiàn)轉(zhuǎn)換。

情況1:__bridge轉(zhuǎn)換

/*
MRC代碼下,將id變量直接強(qiáng)制轉(zhuǎn)換void*正常,但ARC下報(bào)錯(cuò)
id obj = [[NSObject alloc] init];
void *p = obj;
*/
 
//ARC下代碼使用__bridge實(shí)現(xiàn)單純的賦值轉(zhuǎn)換
id obj = [[NSObject alloc] init];
void *p = (__bridge void *)(obj);

__bridge可實(shí)現(xiàn)Objective-C對(duì)象和Core Foundation對(duì)象的相互轉(zhuǎn)換;但是其安全性與賦值給__unsafe_unretained修飾符相近,甚至?xí)汀H绻芾頃r(shí)不注意賦值對(duì)象的所有者,就容易產(chǎn)生野指針錯(cuò)誤導(dǎo)致程序崩潰。

情況2:__bridge_retained轉(zhuǎn)換

id obj = [[NSObject alloc] init];
void *p = (__bridge_retained void *)(obj);
    
/*相當(dāng)于MRC代碼:
id obj = [[NSObject alloc] init];
void *p = obj;
[(id)p retain];
*/

__bridge_retained轉(zhuǎn)換可使要轉(zhuǎn)換的變量也持有所賦值的對(duì)象。此操作類似于retain。上述代碼中變量obj和變量p同時(shí)持有對(duì)象。

情況3:__bridge_transfer轉(zhuǎn)換

id obj = (__bridge_transfer id)p;
    
/*相當(dāng)于MRC代碼:
id obj = id(p)
[obj retain];
[(id)p release];
*/

__bridge_transfer轉(zhuǎn)換提供與__bridge_retained相反的動(dòng)作,被轉(zhuǎn)換的變量所持有的對(duì)象在該變量被賦值給轉(zhuǎn)換的目標(biāo)后隨之釋放。此操作與release相似。

五、內(nèi)存泄漏問題

內(nèi)存泄露就是本該廢棄的對(duì)象在超出其生命周期后繼續(xù)存在。導(dǎo)致系統(tǒng)內(nèi)存浪費(fèi)、程序運(yùn)行速度減慢甚至系統(tǒng)崩潰等嚴(yán)重后果;

總結(jié)常見的內(nèi)存泄露的異常情況如下:

  1. AFNetworking循環(huán)引用(未使用單例或者沒有調(diào)用銷毀NSURLSession的方法;
  2. Block循環(huán)引用
  3. delegate循環(huán)引用
  4. NSTimer循環(huán)引用
  5. 創(chuàng)建的非OC對(duì)象內(nèi)存,在使用完畢后未手動(dòng)釋放;
  6. 循環(huán)操作創(chuàng)建大量臨時(shí)對(duì)象,導(dǎo)致內(nèi)存導(dǎo)致內(nèi)存暴漲;
  7. 地圖類處理,使用完畢后未及時(shí)銷毀地圖相關(guān)組件對(duì)象

六、野指針問題

野指針指針就是指向一個(gè)已經(jīng)刪除對(duì)象或者訪問受限內(nèi)存區(qū)域的指針;

注意:野指針不是nil指針,而是指向”垃圾”內(nèi)存(不可用內(nèi)存)的指針;

總結(jié)ARC下常見的野指針異常情況如下:

ARC野指針.png

參考鏈接:

1.Foundation對(duì)象與Core Foundation對(duì)象間的轉(zhuǎn)換
2.iOS幾種容易忽略的內(nèi)存泄漏方式
3.ARC下野指針常見寫法

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
禁止轉(zhuǎn)載,如需轉(zhuǎn)載請(qǐng)通過簡(jiǎn)信或評(píng)論聯(lián)系作者。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,837評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,196評(píng)論 3 414
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,688評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,654評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,456評(píng)論 6 406
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,955評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,044評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,195評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,725評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,608評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,802評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,318評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,048評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,422評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,673評(píng)論 1 281
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,424評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,762評(píng)論 2 372