《編寫高質(zhì)量iOS與OS X代碼的52個(gè)有效方法》--第五章 第33條
(ps:此乃讀書筆記,加深記憶,僅供大家參考)
第33條 以弱引用避免保留環(huán)
對(duì)象圖里經(jīng)常會(huì)出現(xiàn)一種情況,就是幾個(gè)對(duì)象都以某種方式互相吸引,從而形成“環(huán)”(cycle)。由于Objective-C內(nèi)存管理模型使用引用計(jì)數(shù)架構(gòu),所以這種情況通常會(huì)泄漏內(nèi)存,因?yàn)樽詈鬀]有別的東西會(huì)引用環(huán)中的對(duì)象。這樣的話,環(huán)里的對(duì)象就無法為外界所訪問了,但對(duì)象之間尚有引用,這些引用使得它們都能繼續(xù)存活下去,而不會(huì)為系統(tǒng)所回收。
@class EOCClassA, EOCClassB;
@interface EOCClassA : NSObject
@property (nonatomic, strong) EOCClassB *other;
@end
@interface EOCClassB : NSObject
@property (nonatomic, strong) EOCClassA *other;
@end
保留環(huán)會(huì)導(dǎo)致內(nèi)存泄漏。如果只剩一個(gè)引用還指向保留環(huán)中的實(shí)例,而現(xiàn)在又把這個(gè)引用移除,那么整個(gè)保留環(huán)就泄露了。
避免保留環(huán)的最佳方式就是弱引用。這種引用經(jīng)常用來表示“非擁有關(guān)系”(nonowning relationship)。將屬性聲明為unsafe_unretained即可。
@class EOCClassA, EOCClassB;
@interface EOCClassA : NSObject
@property (nonatomic, strong) EOCClassB *other;
@end
@interface EOCClassB : NSObject
@property (nonatomic, unsafe_unretained) EOCClassA *other;
@end
屬性特質(zhì)(attribute)中的unsafe_unretained一詞表明,屬性值可能不安全,而且不歸此實(shí)例所擁有。如果系統(tǒng)已經(jīng)把屬性所指的那個(gè)對(duì)象回收了,那么在其上調(diào)用方法可能會(huì)使應(yīng)用程序崩潰。由于本對(duì)象并不保留屬性對(duì)象,因此其有可能為系統(tǒng)所回收。
Objective-C中還有一項(xiàng)與ARC相伴的運(yùn)行期特性,可以令開發(fā)者安全使用弱引用:這就是weak屬性特質(zhì),它與unsafe_unretained的作用完全相同。然而,只要系統(tǒng)把屬性回收,屬性值就會(huì)自動(dòng)設(shè)為nil。(ps:而對(duì)nil發(fā)送消息并不會(huì)使程序崩潰)
當(dāng)指向EOCClassA實(shí)例的引用移除后,unsafe_unretained屬性仍然指向那個(gè)已經(jīng)回收的實(shí)例,而weak屬性則指向nil。
一般來說,如果不擁有某對(duì)象,那就不要保留它。這條規(guī)則對(duì)collection例外,collection雖然并不直接擁有其內(nèi)容,但是它要代表自己所屬的那個(gè)對(duì)象來保留這些元素。有時(shí),對(duì)象中的引用會(huì)指向另外一個(gè)并不歸自己所擁有的對(duì)象,比如Delegate模式就是這樣。
要點(diǎn)
- 將某些引用設(shè)為weak,可避免出現(xiàn)“保留環(huán)”。
- weak引用可以自動(dòng)清空,也可以不自動(dòng)清空。自動(dòng)清空(autonilling)是隨著ARC而引入的新特性,由運(yùn)行期系統(tǒng)來實(shí)現(xiàn)。在具備自動(dòng)清空功能的弱引用上,可以隨意讀取其數(shù)據(jù),因?yàn)檫@種引用不會(huì)指向已經(jīng)回收過得對(duì)象。