49、在上文5(1)中提到:當(dāng)NSObject對(duì)象的retainCount減為0之后,就不要再去打印它的retainCount了,有可能導(dǎo)致crash。
為了驗(yàn)證這個(gè)說(shuō)法,可以通過(guò)初始化一個(gè)對(duì)象并釋放它,然后多次打印這個(gè)對(duì)象的retainCount來(lái)測(cè)試。
測(cè)試結(jié)果有很多種情況,取3種情況展示如下:
可以發(fā)現(xiàn):再對(duì)象的retainCount減為0之后,再去打印它的retainCount,程序會(huì)隨機(jī)在某個(gè)打印處crash。
對(duì)象被釋放了還去訪問(wèn)它導(dǎo)致crash這個(gè)很好理解,但是為什么在[sth release]之后,這個(gè)NSObject的retainCount還是1呢?這是因?yàn)楫?dāng)retainCount為1的時(shí)候,再執(zhí)行release的話對(duì)象就該被釋放了,那么此時(shí)系統(tǒng)就會(huì)選擇直接將對(duì)象釋放掉(這樣retainCount自然也就沒(méi)有了),而不會(huì)多花費(fèi)一次寫內(nèi)存的操作去修改它的retainCount為0。
所以在retainCount為1的時(shí)候執(zhí)行了release之后,再去訪問(wèn)它的retainCount的時(shí)候,就只有兩種結(jié)果:要么得到的retainCount為1,要么直接因?yàn)閼覓熘羔槍?dǎo)致EXC_BAD_ACCESS。
另外,可以注意到,程序crash的位置是不固定的,由此也可以猜測(cè):系統(tǒng)釋放對(duì)象是異步執(zhí)行的。
50、在MRC環(huán)境下,如果在一個(gè)有對(duì)象創(chuàng)建的循環(huán)外嵌套了@autoreleasepool塊,當(dāng)循環(huán)的次數(shù)多了,有可能會(huì)出現(xiàn)這些創(chuàng)建的對(duì)象直到@autoreleasepool塊執(zhí)行完才釋放的情況,如下:
這時(shí)候如果將@autoreleasepool塊移動(dòng)到for循環(huán)的內(nèi)部,每一次for循環(huán)都創(chuàng)建一個(gè)@autoreleasepool塊,那么for循環(huán)內(nèi)創(chuàng)建的對(duì)象就會(huì)被及時(shí)地釋放了,如下圖:
在ARC環(huán)境下則不會(huì)存在這個(gè)問(wèn)題,不管使不使用@autoreleasepool塊,for循環(huán)內(nèi)創(chuàng)建的對(duì)象都會(huì)被及時(shí)地釋放:
51、關(guān)于NSString的retainCount問(wèn)題:
(1)、NSString的retainCount并不總是正確的,在不同的構(gòu)造方法下,可能會(huì)得出不同的retainCount。
這是因?yàn)榫幾g器有些情況下不會(huì)把NSString對(duì)象納入內(nèi)存管理的范疇,在這種情況下編譯器會(huì)把NSString對(duì)象當(dāng)成字符串常量來(lái)處理,就不會(huì)有retainCount了。甚至根據(jù)編譯器的不同,完全相同的代碼也有可能得出不同的retainCount。
這是完全視編譯器而定的。
(2)、但是也可以測(cè)試一下,NSString幾個(gè)常用的構(gòu)造方法會(huì)怎樣影響retainCount。使用以下代碼來(lái)測(cè)試:
輸出結(jié)果如下:
(3)、可以看到:
①、使用直接賦值、init方法或者initWithString:方法,retainCount都是-1,說(shuō)明這種情況下NSString對(duì)象就被當(dāng)做字符串常量來(lái)處理了。同時(shí)可以注意到直接賦值和使用initWithString:方法得到的NSString對(duì)象的地址是相同的,進(jìn)一步說(shuō)明它們被當(dāng)做同一個(gè)字符串常量處理了;
②、比較詭異的是initWithFormat:和stringWithFormat:這兩種構(gòu)造方法,在使用漢字或者使用英文且字符較長(zhǎng)的時(shí)候,編譯器會(huì)將它當(dāng)做正常對(duì)象來(lái)處理,會(huì)有retainCount,當(dāng)英文字符較短時(shí),編譯器會(huì)將它當(dāng)做字符串常量來(lái)處理;
(4)、在這里只是做一個(gè)測(cè)試,其實(shí)關(guān)注retainCount是不符合內(nèi)存管理4個(gè)規(guī)則的管理思想的,進(jìn)行內(nèi)存管理的時(shí)候關(guān)注重點(diǎn)應(yīng)當(dāng)是持有與釋放對(duì)象,而不是去關(guān)注有多少指針持有了這個(gè)對(duì)象(retainCount)。