鍍金池/ 教程/ iOS/ Swift解決閉包引起的循環(huán)強(qiáng)引用
Swift模式
Swift溢出運(yùn)算符
Swift連接多層鏈接
Swift類聲明
Swift類型特性
Swift解決閉包引起的循環(huán)強(qiáng)引用
Swift函數(shù)參數(shù)名稱
Swift下標(biāo)
Swift嵌套函數(shù)
Swift字符串插值
swift數(shù)值型字面量
Swift條件語句
Swift實(shí)例方法
Swift定義一個(gè)基類
Swift協(xié)議聲明
Swift比較字符串
Swift空白與注釋
Swift浮點(diǎn)數(shù)
Swift區(qū)間運(yùn)算符
Swift閉包
Swift數(shù)據(jù)類型
Swift Unicode
Swift高級(jí)運(yùn)算符
Swift擴(kuò)展語法
Swift自動(dòng)引用計(jì)數(shù)實(shí)踐
Swift下標(biāo)腳本用法
Swift捕獲值
Swift for循環(huán)
Swift 繼承
Swift常量
Swift通過閉包和函數(shù)來設(shè)置屬性的默認(rèn)值
Swift比較運(yùn)算符
Swift循環(huán)語句
Swift析構(gòu)聲明
Swift防止重寫
Swift嵌套類型
Swift if...else語句
Swift集合的可變性
Swift訪問控制
Swift break語句
Swift函數(shù)類型
Swift原始值
Swift字符串是值類型
Swift For語句
Swift命名類型參數(shù)
Swift類型安全和類型推斷
Swift邏輯運(yùn)算符
Swift決策
Swift字符串和字符
Swift閉包是引用類型
Swift三元條件運(yùn)算
Swift泛型形參子句
Swift突變方法要求
Swift函數(shù)的定義與調(diào)用
Swift Any和AnyObject類型轉(zhuǎn)換
Swift泛型類型
Swift閉包引起的循環(huán)強(qiáng)引用
Swift 屬性
Swift優(yōu)先級(jí)和結(jié)合性
Swift主表達(dá)式
Swift重寫
Swift運(yùn)算術(shù)語
Swift協(xié)議合成類型
Swift構(gòu)造器聲明
Swift 擴(kuò)展
Swift 反初始化
Swift函數(shù)參數(shù)與返回值
Swift結(jié)構(gòu)體
Swift聲明特性
Swift語句
Swift引入聲明
Swift ARC自動(dòng)引用計(jì)數(shù)
Swift類型轉(zhuǎn)換
Swift協(xié)議類型
Swift繼承
Swift運(yùn)算符函數(shù)
Swift集合中的協(xié)議類型
Swift While循環(huán)
Swift協(xié)議
Swift變量聲明
Swift方法擴(kuò)展
Swift通過可選鏈調(diào)用屬性
Swift類和結(jié)構(gòu)體對(duì)比
Swift Switch語句
Swift分支語句
Swift協(xié)議的語法
Swift值類型的構(gòu)造器代理
Swift協(xié)議的繼承
Swift算術(shù)運(yùn)算符
Swift在擴(kuò)展中添加協(xié)議成員
Swift字符串可變性
Swift類型注解
Swift擴(kuò)展聲明
Swift自動(dòng)引用計(jì)數(shù)
Swift類是引用類型
Swift析構(gòu)過程
Swift存儲(chǔ)屬性
Swift全局變量和局部變量
Swift下標(biāo)腳本聲明
Swift類和結(jié)構(gòu)體
Swift數(shù)值型類型轉(zhuǎn)換
Swift類型別名
Swift 協(xié)議
swift分號(hào)
Swift賦值表達(dá)式
Swift計(jì)算字符數(shù)量
Swift定制化構(gòu)造過程
Swift運(yùn)算符優(yōu)先級(jí)
Swift可選鏈
Swift標(biāo)識(shí)符
Swift泛型所解決的問題
Swift二元表達(dá)式
Swift閉包
Swift匹配枚舉值和Switch語句
Swift使用字符
Swift斷言
Swift類型的別名聲明
Swift可選鏈可替代強(qiáng)制解析
Swift后綴表達(dá)式
Swift帶標(biāo)簽的語句
Swift屬性要求
Swift連接字符串和字符
Swift自定義運(yùn)算符
Swift類型轉(zhuǎn)換運(yùn)算符
Swift閉包表達(dá)式
Swift枚舉語法
Swift類實(shí)例之間的循環(huán)強(qiáng)引用
Swift字符
Swift復(fù)合賦值
Swift析構(gòu)過程原理
Swift枚舉
Swift結(jié)構(gòu)體聲明
Swift do...while循環(huán)
Swift運(yùn)算符
Swift使用可選鏈調(diào)用子腳本
Swift控制轉(zhuǎn)移語句
Swift前綴表達(dá)式
Swift表達(dá)式模式
Swift開發(fā)環(huán)境設(shè)置
Swift類型約束
Swift while循環(huán)
Swift元組類型
Swift枚舉用例模式
Swift默認(rèn)構(gòu)造器
Swift 方法
Swift委托(代理)模式
Swift類型轉(zhuǎn)換模式
Swift類型方法
Swift方法要求
Swift位運(yùn)算符
Swift下標(biāo)腳本
Swift可選類型
Swift聲明
Swift類型標(biāo)識(shí)符
Swift元組
Swift字符串
Swift枚舉聲明
Swift屬性
Swift模塊范圍
Swift泛型
Swift類的繼承和構(gòu)造過程
Swift定義一個(gè)類層次作為例子
Swift構(gòu)造過程
Swift泛型函數(shù)
Swift快速入門
Swift注釋
Swift基礎(chǔ)
Swift下標(biāo)腳本選項(xiàng)
Swift數(shù)組類型
Swift關(guān)聯(lián)類型
Swift集合(Collection)類型的賦值和拷貝行為
Swift類型參數(shù)
Swift 類型轉(zhuǎn)換
Swift下標(biāo)腳本語法
Swift計(jì)算屬性
Swift方法
Swift檢查類型
Swift標(biāo)識(shí)符模式
Swift比較運(yùn)算
Swift 泛型
Swift基本語法
Swift賦值運(yùn)算符
Swift布爾值
Swift位運(yùn)算符
Swift函數(shù)
Swift初始化空字符串
Swift為可選鏈定義模型類
Swift元組模式
Swift通過可選鏈調(diào)用方法
Swift if...else if...else語句
Swift通過擴(kuò)展補(bǔ)充協(xié)議聲明
Swift可選類型(命名型類型)
Swift類型推斷
Swift continue語句
Swift泛型參數(shù)
Swift協(xié)議合成
Swift For循環(huán)
Swift控制流
Swift結(jié)構(gòu)體和枚舉是值類型
Swift函數(shù)類型(參數(shù)類型和返回值類型)
Swift自動(dòng)引用計(jì)數(shù)的工作機(jī)制
Swift fallthrough語句
Swift字符串字面量
Swift運(yùn)算符聲明
Swift函數(shù)
Swift類
Swift類型屬性
Swift表達(dá)式
Swift范圍運(yùn)算符
Swift邏輯運(yùn)算
Swift賦值運(yùn)算符
Swift if語句
Swift數(shù)組
Swift枚舉
Swift循環(huán)
Swift擴(kuò)展
Swift檢驗(yàn)協(xié)議的一致性
Swift解決實(shí)例之間的循環(huán)強(qiáng)引用
Swift整數(shù)
Swift函數(shù)聲明
Swift嵌套類型擴(kuò)展
Swift 下標(biāo)
Swift可選協(xié)議要求
Swift子類生成
Swift變量
Swift計(jì)算型屬性
Swift 可選鏈
Swift嵌套類型實(shí)例
Swift類型繼承子句
Swift教程
Swift析構(gòu)函數(shù)操作
Swift存儲(chǔ)型屬性的初始賦值
Swift類和結(jié)構(gòu)體的選擇
Swift類型
Swift數(shù)組
Swift相關(guān)值
Swift向下轉(zhuǎn)型
Swift特性
Swift屬性監(jiān)視器
Swift語法結(jié)構(gòu)
Swift尾隨閉包
Swift字典
Swift代碼塊
Swift其它運(yùn)算符
Swift嵌套類型的引用
Swift Where語句
Swift關(guān)鍵字
Swift數(shù)值運(yùn)算
Swift集合類型 (Collection Types)
Swift通配符模式
Swift運(yùn)算符
Swift字面量
Swift常量聲明
Swift元類型
Swift字面量
Swift初始化
Swift鏈接可選返回值的方法
Swift值綁定模式
Swift隱式解析可選類型
Swift嵌套 if 語句
Swift大寫和小寫字符串
Swift構(gòu)造器
Swift for-in循環(huán)
Swift常量和變量

Swift解決閉包引起的循環(huán)強(qiáng)引用

解決閉包引起的循環(huán)強(qiáng)引用

在定義閉包時(shí)同時(shí)定義捕獲列表作為閉包的一部分,通過這種方式可以解決閉包和類實(shí)例之間的循環(huán)強(qiáng)引用。捕獲列表定義了閉包體內(nèi)捕獲一個(gè)或者多個(gè)引用類型的規(guī)則。跟解決兩個(gè)類實(shí)例間的循環(huán)強(qiáng)引用一樣,聲明每個(gè)捕獲的引用為弱引用或無主引用,而不是強(qiáng)引用。應(yīng)當(dāng)根據(jù)代碼關(guān)系來決定使用弱引用還是無主引用。


注意:
Swift 有如下要求:只要在閉包內(nèi)使用self的成員,就要用self.someProperty或者self.someMethod(而不只是somePropertysomeMethod)。這提醒你可能會(huì)不小心就捕獲了self。
 

定義捕獲列表

捕獲列表中的每個(gè)元素都是由weak或者unowned關(guān)鍵字和實(shí)例的引用(如selfsomeInstance)成對(duì)組成。每一對(duì)都在方括號(hào)中,通過逗號(hào)分開。

捕獲列表放置在閉包參數(shù)列表和返回類型之前:

@lazy var someClosure: (Int, String) -> String = {
    [unowned self] (index: Int, stringToProcess: String) -> String in
    // closure body goes here
}

如果閉包沒有指定參數(shù)列表或者返回類型,則可以通過上下文推斷,那么可以捕獲列表放在閉包開始的地方,跟著是關(guān)鍵字in

@lazy var someClosure: () -> String = {
    [unowned self] in
    // closure body goes here
}

弱引用和無主引用

當(dāng)閉包和捕獲的實(shí)例總是互相引用時(shí)并且總是同時(shí)銷毀時(shí),將閉包內(nèi)的捕獲定義為無主引用。

相反的,當(dāng)捕獲引用有時(shí)可能會(huì)是nil時(shí),將閉包內(nèi)的捕獲定義為弱引用。弱引用總是可選類型,并且當(dāng)引用的實(shí)例被銷毀后,弱引用的值會(huì)自動(dòng)置為nil。這使我們可以在閉包內(nèi)檢查它們是否存在。


注意:
如果捕獲的引用絕對(duì)不會(huì)置為nil,應(yīng)該用無主引用,而不是弱引用。
 

前面的HTMLElement例子中,無主引用是正確的解決循環(huán)強(qiáng)引用的方法。這樣編寫HTMLElement類來避免循環(huán)強(qiáng)引用:

class HTMLElement {

    let name: String
    let text: String?

    @lazy var asHTML: () -> String = {
        [unowned self] in
        if let text = self.text {
            return "<\(self.name)>\(text)</\(self.name)>"
        } else {
            return "<\(self.name) />"
        }
    }

    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }

    deinit {
        println("\(name) is being deinitialized")
    }

}

上面的HTMLElement實(shí)現(xiàn)和之前的實(shí)現(xiàn)一致,只是在asHTML閉包中多了一個(gè)捕獲列表。這里,捕獲列表是[unowned self],表示“用無主引用而不是強(qiáng)引用來捕獲self”。

和之前一樣,我們可以創(chuàng)建并打印HTMLElement實(shí)例:

var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
println(paragraph!.asHTML())
// prints "<p>hello, world</p>"

使用捕獲列表后引用關(guān)系如下圖所示:

這一次,閉包以無主引用的形式捕獲self,并不會(huì)持有HTMLElement實(shí)例的強(qiáng)引用。如果將paragraph賦值為nil,HTMLElement實(shí)例將會(huì)被銷毀,并能看到它的析構(gòu)函數(shù)打印出的消息。

paragraph = nil
// prints "p is being deinitialized"