鍍金池/ 教程/ iOS/ Swift解決實(shí)例之間的循環(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條件語(yǔ)句
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ò)展語(yǔ)法
Swift自動(dòng)引用計(jì)數(shù)實(shí)踐
Swift下標(biāo)腳本用法
Swift捕獲值
Swift for循環(huán)
Swift 繼承
Swift常量
Swift通過(guò)閉包和函數(shù)來(lái)設(shè)置屬性的默認(rèn)值
Swift比較運(yùn)算符
Swift循環(huán)語(yǔ)句
Swift析構(gòu)聲明
Swift防止重寫
Swift嵌套類型
Swift if...else語(yǔ)句
Swift集合的可變性
Swift訪問(wèn)控制
Swift break語(yǔ)句
Swift函數(shù)類型
Swift原始值
Swift字符串是值類型
Swift For語(yǔ)句
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ù)語(yǔ)
Swift協(xié)議合成類型
Swift構(gòu)造器聲明
Swift 擴(kuò)展
Swift 反初始化
Swift函數(shù)參數(shù)與返回值
Swift結(jié)構(gòu)體
Swift聲明特性
Swift語(yǔ)句
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通過(guò)可選鏈調(diào)用屬性
Swift類和結(jié)構(gòu)體對(duì)比
Swift Switch語(yǔ)句
Swift分支語(yǔ)句
Swift協(xié)議的語(yǔ)法
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)過(guò)程
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)造過(guò)程
Swift運(yùn)算符優(yōu)先級(jí)
Swift可選鏈
Swift標(biāo)識(shí)符
Swift泛型所解決的問(wèn)題
Swift二元表達(dá)式
Swift閉包
Swift匹配枚舉值和Switch語(yǔ)句
Swift使用字符
Swift斷言
Swift類型的別名聲明
Swift可選鏈可替代強(qiáng)制解析
Swift后綴表達(dá)式
Swift帶標(biāo)簽的語(yǔ)句
Swift屬性要求
Swift連接字符串和字符
Swift自定義運(yùn)算符
Swift類型轉(zhuǎn)換運(yùn)算符
Swift閉包表達(dá)式
Swift枚舉語(yǔ)法
Swift類實(shí)例之間的循環(huán)強(qiáng)引用
Swift字符
Swift復(fù)合賦值
Swift析構(gòu)過(guò)程原理
Swift枚舉
Swift結(jié)構(gòu)體聲明
Swift do...while循環(huán)
Swift運(yùn)算符
Swift使用可選鏈調(diào)用子腳本
Swift控制轉(zhuǎn)移語(yǔ)句
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)造過(guò)程
Swift定義一個(gè)類層次作為例子
Swift構(gòu)造過(guò)程
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)腳本語(yǔ)法
Swift計(jì)算屬性
Swift方法
Swift檢查類型
Swift標(biāo)識(shí)符模式
Swift比較運(yùn)算
Swift 泛型
Swift基本語(yǔ)法
Swift賦值運(yùn)算符
Swift布爾值
Swift位運(yùn)算符
Swift函數(shù)
Swift初始化空字符串
Swift為可選鏈定義模型類
Swift元組模式
Swift通過(guò)可選鏈調(diào)用方法
Swift if...else if...else語(yǔ)句
Swift通過(guò)擴(kuò)展補(bǔ)充協(xié)議聲明
Swift可選類型(命名型類型)
Swift類型推斷
Swift continue語(yǔ)句
Swift泛型參數(shù)
Swift協(xié)議合成
Swift For循環(huán)
Swift控制流
Swift結(jié)構(gòu)體和枚舉是值類型
Swift函數(shù)類型(參數(shù)類型和返回值類型)
Swift自動(dòng)引用計(jì)數(shù)的工作機(jī)制
Swift fallthrough語(yǔ)句
Swift字符串字面量
Swift運(yùn)算符聲明
Swift函數(shù)
Swift類
Swift類型屬性
Swift表達(dá)式
Swift范圍運(yùn)算符
Swift邏輯運(yùn)算
Swift賦值運(yùn)算符
Swift if語(yǔ)句
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語(yǔ)法結(jié)構(gòu)
Swift尾隨閉包
Swift字典
Swift代碼塊
Swift其它運(yùn)算符
Swift嵌套類型的引用
Swift Where語(yǔ)句
Swift關(guān)鍵字
Swift數(shù)值運(yùn)算
Swift集合類型 (Collection Types)
Swift通配符模式
Swift運(yùn)算符
Swift字面量
Swift常量聲明
Swift元類型
Swift字面量
Swift初始化
Swift鏈接可選返回值的方法
Swift值綁定模式
Swift隱式解析可選類型
Swift嵌套 if 語(yǔ)句
Swift大寫和小寫字符串
Swift構(gòu)造器
Swift for-in循環(huán)
Swift常量和變量

Swift解決實(shí)例之間的循環(huán)強(qiáng)引用

解決實(shí)例之間的循環(huán)強(qiáng)引用

Swift 提供了兩種辦法用來(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)?code style="box-sizing: border-box; -webkit-tap-highlight-color: transparent; -webkit-font-smoothing: antialiased; font-family: Monaco, Menlo, Consolas, 'Courier New', monospace; font-size: 14px; padding: 0px 5px; color: rgb(199, 37, 78); background-color: rgb(248, 248, 248); white-space: nowrap; border-top-left-radius: 3px; border-top-right-radius: 3px; border-bottom-right-radius: 3px; border-bottom-left-radius: 3px; margin: 0px 2px; border: 1px solid rgb(234, 234, 234);">nil的實(shí)例使用弱引用。相反的,對(duì)于初始化賦值后再也不會(huì)被賦值為nil的實(shí)例,使用無(wú)主引用。

弱引用

弱引用不會(huì)牢牢保持住引用的實(shí)例,并且不會(huì)阻止 ARC 銷毀被引用的實(shí)例。這種行為阻止了引用變?yōu)檠h(huán)強(qiáng)引用。聲明屬性或者變量時(shí),在前面加上weak關(guān)鍵字表明這是一個(gè)弱引用。

在實(shí)例的生命周期中,如果某些時(shí)候引用沒(méi)有值,那么弱引用可以阻止循環(huán)強(qiáng)引用。如果引用總是有值,則可以使用無(wú)主引用,在無(wú)主引用中有描述。在上面Apartment的例子中,一個(gè)公寓的生命周期中,有時(shí)是沒(méi)有“居民”的,因此適合使用弱引用來(lái)解決循環(huán)強(qiáng)引用。


注意:
弱引用必須被聲明為變量,表明其值能在運(yùn)行時(shí)被修改。弱引用不能被聲明為常量。
 

因?yàn)槿跻每梢詻](méi)有值,你必須將每一個(gè)弱引用聲明為可選類型。可選類型是在 Swift 語(yǔ)言中推薦的用來(lái)表示可能沒(méi)有值的類型。

因?yàn)槿跻貌粫?huì)保持所引用的實(shí)例,即使引用存在,實(shí)例也有可能被銷毀。因此,ARC 會(huì)在引用的實(shí)例被銷毀后自動(dòng)將其賦值為nil。你可以像其他可選值一樣,檢查弱引用的值是否存在,你永遠(yuǎn)也不會(huì)遇到被銷毀了而不存在的實(shí)例。

下面的例子跟上面PersonApartment的例子一致,但是有一個(gè)重要的區(qū)別。這一次,Apartmenttenant屬性被聲明為弱引用:

class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
    deinit { println("\(name) is being deinitialized") }
}
class Apartment {
    let number: Int
    init(number: Int) { self.number = number }
    weak var tenant: Person?
    deinit { println("Apartment #\(number) is being deinitialized") }
}

然后跟之前一樣,建立兩個(gè)變量(john和number73)之間的強(qiáng)引用,并關(guān)聯(lián)兩個(gè)實(shí)例:

var john: Person?
var number73: Apartment?

john = Person(name: "John Appleseed")
number73 = Apartment(number: 73)

john!.apartment = number73
number73!.tenant = john

現(xiàn)在,兩個(gè)關(guān)聯(lián)在一起的實(shí)例的引用關(guān)系如下圖所示:

Person實(shí)例依然保持對(duì)Apartment實(shí)例的強(qiáng)引用,但是Apartment實(shí)例只是對(duì)Person實(shí)例的弱引用。這意味著當(dāng)你斷開john變量所保持的強(qiáng)引用時(shí),再也沒(méi)有指向Person實(shí)例的強(qiáng)引用了:

由于再也沒(méi)有指向Person實(shí)例的強(qiáng)引用,該實(shí)例會(huì)被銷毀:

john = nil
// prints "John Appleseed is being deinitialized"

唯一剩下的指向Apartment實(shí)例的強(qiáng)引用來(lái)自于變量number73。如果你斷開這個(gè)強(qiáng)引用,再也沒(méi)有指向Apartment實(shí)例的強(qiáng)引用了:

由于再也沒(méi)有指向Apartment實(shí)例的強(qiáng)引用,該實(shí)例也會(huì)被銷毀:

number73 = nil
// prints "Apartment #73 is being deinitialized"

上面的兩段代碼展示了變量johnnumber73在被賦值為nil后,Person實(shí)例和Apartment實(shí)例的析構(gòu)函數(shù)都打印出“銷毀”的信息。這證明了引用循環(huán)被打破了。

無(wú)主引用

和弱引用類似,無(wú)主引用不會(huì)牢牢保持住引用的實(shí)例。和弱引用不同的是,無(wú)主引用是永遠(yuǎn)有值的。因此,無(wú)主引用總是被定義為非可選類型(non-optional type)。你可以在聲明屬性或者變量時(shí),在前面加上關(guān)鍵字unowned表示這是一個(gè)無(wú)主引用。

由于無(wú)主引用是非可選類型,你不需要在使用它的時(shí)候?qū)⑺归_。無(wú)主引用總是可以被直接訪問(wèn)。不過(guò) ARC 無(wú)法在實(shí)例被銷毀后將無(wú)主引用設(shè)為nil,因?yàn)榉强蛇x類型的變量不允許被賦值為nil。


注意:
如果你試圖在實(shí)例被銷毀后,訪問(wèn)該實(shí)例的無(wú)主引用,會(huì)觸發(fā)運(yùn)行時(shí)錯(cuò)誤。使用無(wú)主引用,你必須確保引用始終指向一個(gè)未銷毀的實(shí)例。
還需要注意的是如果你試圖訪問(wèn)實(shí)例已經(jīng)被銷毀的無(wú)主引用,程序會(huì)直接崩潰,而不會(huì)發(fā)生無(wú)法預(yù)期的行為。所以你應(yīng)當(dāng)避免這樣的事情發(fā)生。
 

下面的例子定義了兩個(gè)類,上一篇:Swift變量聲明下一篇:Swift Any和AnyObject類型轉(zhuǎn)換