注意
引用計(jì)數(shù)只應(yīng)用于類(lèi)的實(shí)例狠鸳。結(jié)構(gòu)體和枚舉是值類(lèi)型悯嗓,不是引用類(lèi)型,沒(méi)有通過(guò)引用存儲(chǔ)和傳遞脯厨。
ARC的工作機(jī)制
每次你創(chuàng)建一個(gè)類(lèi)的實(shí)例坑质,ARC 會(huì)分配一大塊內(nèi)存來(lái)存儲(chǔ)這個(gè)實(shí)例的信息。這些內(nèi)存中保留有實(shí)例的類(lèi)型信息稼跳,以及該實(shí)例所有存儲(chǔ)屬性值的信息吃沪。
此外,當(dāng)實(shí)例不需要時(shí)票彪,ARC 會(huì)釋放該實(shí)例所占用的內(nèi)存,釋放的內(nèi)存用于其他用途锉屈。這確保類(lèi)實(shí)例當(dāng)它不在需要時(shí)垮耳,不會(huì)一直占用內(nèi)存。
然而终佛,如果 ARC 釋放了正在使用的實(shí)例內(nèi)存俊嗽,那么它將不會(huì)訪(fǎng)問(wèn)實(shí)例的屬性铃彰,或者調(diào)用實(shí)例的方法。確實(shí)牙捉,如果你試圖訪(fǎng)問(wèn)該實(shí)例,你的app很可能會(huì)崩潰芬位。
為了確保使用中的實(shí)例不會(huì)消失带到,ARC 會(huì)跟蹤和計(jì)算當(dāng)前實(shí)例被多少屬性,常量和變量所引用被饿。只要存在對(duì)該類(lèi)實(shí)例的引用,ARC 將不會(huì)釋放該實(shí)例狭握。
為了使這些成為可能论颅,無(wú)論你將實(shí)例分配給屬性,常量或變量嗅辣,它們都會(huì)創(chuàng)建該實(shí)例的強(qiáng)引用澡谭。之所以稱(chēng)之為“強(qiáng)”引用损俭,是因?yàn)樗鼤?huì)將實(shí)例保持住,只要強(qiáng)引用還在杆兵,實(shí)例是不允許被銷(xiāo)毀的琐脏。
解決實(shí)例之間的循環(huán)強(qiáng)引用
Swift 提供了兩種辦法用來(lái)解決你在使用類(lèi)的屬性時(shí)所遇到的循環(huán)強(qiáng)引用問(wèn)題:弱引用( weak reference )和無(wú)主引用( unowned reference )。
弱引用和無(wú)主引用允許循環(huán)引用中的一個(gè)實(shí)例引用另外一個(gè)實(shí)例而不保持強(qiáng)引用日裙。這樣實(shí)例能夠互相引用而不產(chǎn)生循環(huán)強(qiáng)引用昂拂。
對(duì)于生命周期中會(huì)變?yōu)?nil 的實(shí)例使用弱引用。相反格侯,對(duì)于初始化賦值后再也不會(huì)被賦值為 nil 的實(shí)例联四,使用無(wú)主引用。在實(shí)例的生命周期中碎连,當(dāng)引用可能“沒(méi)有值”的時(shí)候,就使用弱引用來(lái)避免循環(huán)引用廉嚼。如同在無(wú)主引用中描述的那樣,如果引用始終有值恐似,則可以使用無(wú)主引用來(lái)代替
弱引用
弱引用不會(huì)對(duì)其引用的實(shí)例保持強(qiáng)引用傍念,因而不會(huì)阻止 ARC 釋放被引用的實(shí)例。這個(gè)特性阻止了引用變?yōu)檠h(huán)強(qiáng)引用双藕。聲明屬性或者變量時(shí)阳仔,在前面加上 weak 關(guān)鍵字表明這是一個(gè)弱引用。
由于弱引用不會(huì)強(qiáng)保持對(duì)實(shí)例的引用嘶摊,所以說(shuō)實(shí)例被釋放了弱引用仍舊引用著這個(gè)實(shí)例也是有可能的评矩。因此,ARC 會(huì)在被引用的實(shí)例被釋放是自動(dòng)地設(shè)置弱引用為 nil 虱颗。由于弱引用需要允許它們的值為 nil 果录,它們一定得是可選類(lèi)型。
你可以檢查弱引用的值是否存在辨萍,就像其他可選項(xiàng)的值一樣返弹,并且你將永遠(yuǎn)不會(huì)遇到“野指針”。
注意
在 ARC 給弱引用設(shè)置 nil 時(shí)不會(huì)調(diào)用屬性觀察者义起。
無(wú)主引用
和弱引用類(lèi)似默终,無(wú)主引用不會(huì)牢牢保持住引用的實(shí)例犁罩。但是不像弱引用两疚,總之,無(wú)主引用假定是永遠(yuǎn)有值的丐巫。因此勺美,無(wú)主引用總是被定義為非可選類(lèi)型。你可以在聲明屬性或者變量時(shí)缎脾,在前面加上關(guān)鍵字 unowned 表示這是一個(gè)無(wú)主引用占卧。
由于無(wú)主引用是非可選類(lèi)型,你不需要在使用它的時(shí)候?qū)⑺归_(kāi)。無(wú)主引用總是可以直接訪(fǎng)問(wèn)耸袜。不過(guò) ARC 無(wú)法在實(shí)例被釋放后將無(wú)主引用設(shè)為 nil 堤框,因?yàn)榉强蛇x類(lèi)型的變量不允許被賦值為 nil 。
注意
如果你試圖在實(shí)例的被釋放后訪(fǎng)問(wèn)無(wú)主引用蜈抓,那么你將觸發(fā)運(yùn)行時(shí)錯(cuò)誤沟使。只有在你確保引用會(huì)一直引用實(shí)例的時(shí)候才使用無(wú)主引用。
還要注意的是腊嗡,如果你試圖訪(fǎng)問(wèn)引用的實(shí)例已經(jīng)被釋放了的無(wú)主引用燕少,Swift 會(huì)確保程序直接崩潰。你不會(huì)因此而遭遇無(wú)法預(yù)期的行為客们。所以你應(yīng)當(dāng)避免這樣的事情發(fā)生。
閉包的循環(huán)強(qiáng)引用
上面我們看到了循環(huán)強(qiáng)引用是如何在兩個(gè)實(shí)例屬性互相保持對(duì)方的強(qiáng)引用時(shí)產(chǎn)生的恒傻,還知道了如何用弱引用和無(wú)主引用來(lái)打破這些循環(huán)強(qiáng)引用。
循環(huán)強(qiáng)引用還會(huì)出現(xiàn)在你把一個(gè)閉包分配給類(lèi)實(shí)例屬性的時(shí)候湿痢,并且這個(gè)閉包中又捕獲了這個(gè)實(shí)例扑庞。捕獲可能發(fā)生于這個(gè)閉包函數(shù)體中訪(fǎng)問(wèn)了實(shí)例的某個(gè)屬性,比如 self.someProperty 臀规,或者這個(gè)閉包調(diào)用了一個(gè)實(shí)例的方法栅隐,例如 self.someMethod() 。這兩種情況都導(dǎo)致了閉包 “捕獲”了 self 谨究,從而產(chǎn)生了循環(huán)強(qiáng)引用泣棋。
循環(huán)強(qiáng)引用的產(chǎn)生,是因?yàn)殚]包和類(lèi)相似鸯屿,都是引用類(lèi)型把敢。當(dāng)你把閉包賦值給了一個(gè)屬性,你實(shí)際上是把一個(gè)引用賦值給了這個(gè)閉包婶恼。實(shí)質(zhì)上柏副,這跟之前上面的問(wèn)題是一樣的——兩個(gè)強(qiáng)引用讓彼此一直有效〖焯担總之锨推,和兩個(gè)類(lèi)實(shí)例不同公壤,這次一個(gè)是類(lèi)實(shí)例和一個(gè)閉包互相引用厦幅。
解決閉包的循環(huán)強(qiáng)引用
你可以通過(guò)定義捕獲列表作為閉包的定義來(lái)解決在閉包和類(lèi)實(shí)例之間的循環(huán)強(qiáng)引用慨飘。捕獲列表定義了當(dāng)在閉包體里捕獲一個(gè)或多個(gè)引用類(lèi)型的規(guī)則。正如在兩個(gè)類(lèi)實(shí)例之間的循環(huán)強(qiáng)引用休弃,聲明每個(gè)捕獲的引用為引用或無(wú)主引用而不是強(qiáng)引用圈膏。應(yīng)當(dāng)根據(jù)代碼關(guān)系來(lái)決定使用弱引用還是無(wú)主引用。
注意
Swift 要求你在閉包中引用self成員時(shí)使用 self.someProperty 或者 self.someMethod (而不只是 someProperty 或 someMethod )丈甸。這有助于提醒你可能會(huì)一不小心就捕獲了 self 尿褪。
定義捕獲列表
捕獲列表中的每一項(xiàng)都由 weak 或 unowned 關(guān)鍵字與類(lèi)實(shí)例的引用(如 self )或初始化過(guò)的變量(如 delegate = self.delegate! )成對(duì)組成杖玲。這些項(xiàng)寫(xiě)在方括號(hào)中用逗號(hào)分開(kāi)。
把捕獲列表放在形式參數(shù)和返回類(lèi)型前邊天揖,如果它們存在的話(huà):
lazy var someClosure: (Int, String) -> String = {
[unowned self, weak delegate = self.delegate!] (index: Int, stringToProcess: String) -> String in
// closure body goes here
}
lazy var someClosure: (Int, String) -> String = {
[unowned self, weak delegate = self.delegate!] (index: Int, stringToProcess: String) -> String in
// closure body goes here
}
如果閉包沒(méi)有指明形式參數(shù)列表或者返回類(lèi)型今膊,是因?yàn)樗鼈儠?huì)通過(guò)上下文推斷伞剑,那么就把捕獲列表放在關(guān)鍵字 in 前邊,閉包最開(kāi)始的地方:
lazy var someClosure: () -> String = {
[unowned self, weak delegate = self.delegate!] in
// closure body goes here
}
lazy var someClosure: () -> String = {
[unowned self, weak delegate = self.delegate!] in
// closure body goes here
}