自動(dòng)引用計(jì)數(shù)(學(xué)習(xí)筆記)

注意
引用計(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
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市抒倚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌含蓉,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件斟赚,死亡現(xiàn)場(chǎng)離奇詭異差油,居然都是意外死亡蓄喇,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)器紧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)楼眷,“玉大人,你說(shuō)我怎么就攤上這事掌腰≌偶” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,764評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵勺择,是天一觀的道長(zhǎng)省核。 經(jīng)常有香客問(wèn)我昆码,道長(zhǎng),這世上最難降的妖魔是什么旧噪? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,193評(píng)論 1 292
  • 正文 為了忘掉前任脓匿,我火速辦了婚禮,結(jié)果婚禮上日月,老公的妹妹穿的比我還像新娘。我一直安慰自己尺借,他們只是感情好精拟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布蜂绎。 她就那樣靜靜地躺著,像睡著了一般怪瓶。 火紅的嫁衣襯著肌膚如雪践美。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,182評(píng)論 1 299
  • 那天敛滋,我揣著相機(jī)與錄音兴革,去河邊找鬼。 笑死庶艾,一個(gè)胖子當(dāng)著我的面吹牛擎勘,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 40,063評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼蟹地,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼藤为!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起分别,我...
    開(kāi)封第一講書(shū)人閱讀 38,917評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤耘斩,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后括授,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體荚虚,經(jīng)...
    沈念sama閱讀 45,329評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評(píng)論 2 332
  • 正文 我和宋清朗相戀三年梯澜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了晚伙。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片檬某。...
    茶點(diǎn)故事閱讀 39,722評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖民傻,靈堂內(nèi)的尸體忽然破棺而出场斑,到底是詐尸還是另有隱情,我是刑警寧澤喧半,帶...
    沈念sama閱讀 35,425評(píng)論 5 343
  • 正文 年R本政府宣布挺据,位于F島的核電站脖隶,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏产阱。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評(píng)論 3 326
  • 文/蒙蒙 一王暗、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧科汗,春花似錦策肝、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,671評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)膘婶。三九已至,卻和暖如春悬襟,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背逝段。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,825評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工奶躯, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人亿驾。 一個(gè)月前我還...
    沈念sama閱讀 47,729評(píng)論 2 368
  • 正文 我出身青樓嘹黔,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親莫瞬。 傳聞我的和親對(duì)象是個(gè)殘疾皇子儡蔓,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容