Swift教學
Swift快速入門
Swift簡介
Swift基礎
Swift常量和變量
Swift註釋
swift分號
Swift整數
Swift浮點數
Swift類型安全和類型推斷
swift數值型字面量
Swift數值型類型轉換
Swift類型別名
Swift布爾值
Swift元組
Swift可選類型
Swift斷言
Swift基本運算符
Swif哈希集合
Swift運算術語
Swift字典
Swift賦值運算符
Swift數值運算
Swift複合賦值
Swift比較運算
Swift三元條件運算
Swift區間運算符
Swift邏輯運算
Swift字符串和字符
Swift字符串字面量
Swift初始化空字符串
Swift字符串可變性
Swift字符串是值類型
Swift使用字符
Swift計算字符數量
Swift連接字符串和字符
Swift字符串插值
Swift比較字符串
Swift大寫和小寫字符串
Swift Unicode
Swift集合類型 (Collection Types)
Swift數組
Swift字典
Swift集合的可變性
Swift控制流
Swift For循環
Swift While循環
Swift條件語句
Swift控制轉移語句
Swift函數
Swift函數的定義與調用
Swift函數參數與返回值
Swift函數參數名稱
Swift函數類型
Swift嵌套函數
Swift閉包
Swift閉包表達式
Swift尾隨閉包
Swift捕獲值
Swift閉包是引用類型
Swift枚舉
Swift枚舉語法
Swift匹配枚舉值和Switch語句
Swift相關值
Swift原始值
Swift類和結構體
Swift類和結構體對比
Swift結構體和枚舉是值類型
Swift類是引用類型
Swift類和結構體的選擇
Swift集合(Collection)類型的賦值和拷貝行爲
Swift屬性
Swift存儲屬性
Swift計算屬性
Swift屬性監視器
Swift全局變量和局部變量
Swift類型屬性
Swift方法
Swift實例方法
Swift類型方法
Swift下標腳本
Swift下標腳本語法
Swift下標腳本用法
Swift下標腳本選項
Swift繼承
Swift定義一個基類
Swift子類生成
Swift重寫
Swift防止重寫
Swift構造過程
Swift存儲型屬性的初始賦值
Swift定製化構造過程
Swift默認構造器
Swift值類型的構造器代理
Swift類的繼承和構造過程
Swift通過閉包和函數來設置屬性的默認值
Swift析構過程
Swift析構過程原理
Swift析構函數操作
Swift自動引用計數
Swift自動引用計數的工作機制
Swift自動引用計數實踐
Swift類實例之間的循環強引用
Swift解決實例之間的循環強引用
Swift閉包引起的循環強引用
Swift解決閉包引起的循環強引用
Swift可選鏈
Swift可選鏈可替代強制解析
Swift爲可選鏈定義模型類
Swift通過可選鏈調用屬性
Swift通過可選鏈調用方法
Swift使用可選鏈調用子腳本
Swift連接多層鏈接
Swift鏈接可選返回值的方法
Swift類型轉換
Swift定義一個類層次作爲例子
Swift檢查類型
Swift向下轉型
Swift Any和AnyObject類型轉換
Swift嵌套類型
Swift嵌套類型實例
Swift嵌套類型的引用
Swift擴展
Swift擴展語法
Swift計算型屬性
Swift構造器
Swift方法擴展
Swift下標
Swift嵌套類型擴展
Swift協議
Swift協議的語法
Swift屬性要求
Swift方法要求
Swift突變方法要求
Swift協議類型
Swift委託(代理)模式
Swift在擴展中添加協議成員
Swift通過擴展補充協議聲明
Swift集合中的協議類型
Swift協議的繼承
Swift協議合成
Swift檢驗協議的一致性
Swift可選協議要求
Swift泛型
Swift泛型所解決的問題
Swift泛型函數
Swift類型參數
Swift命名類型參數
Swift泛型類型
Swift類型約束
Swift關聯類型
Swift Where語句
Swift高級運算符
Swift位運算符
Swift溢出運算符
Swift優先級和結合性
Swift運算符函數
Swift自定義運算符
Swift語法結構
Swift空白與註釋
Swift標識符
Swift關鍵字
Swift字面量
Swift運算符
Swift類型
Swift類型註解
Swift類型標識符
Swift元組類型
Swift函數類型(參數類型和返回值類型)
Swift數組類型
Swift可選類型(命名型類型)
Swift隱式解析可選類型
Swift協議合成類型
Swift元類型
Swift類型繼承子句
Swift類型推斷
Swift表達式
Swift前綴表達式
Swift二元表達式
Swift賦值表達式
Swift類型轉換運算符
Swift主表達式
Swift後綴表達式
Swift語句
Swift循環語句
Swift For語句
Swift分支語句
Swift帶標籤的語句
Swift聲明
Swift模塊範圍
Swift代碼塊
Swift引入聲明
Swift常量聲明
Swift類型的別名聲明
Swift函數聲明
Swift枚舉聲明
Swift結構體聲明
Swift類聲明
Swift協議聲明
Swift構造器聲明
Swift析構聲明
Swift擴展聲明
Swift下標腳本聲明
Swift運算符聲明
Swift變量聲明
Swift特性
Swift聲明特性
Swift類型特性
Swift模式
Swift通配符模式
Swift標識符模式
Swift值綁定模式
Swift元組模式
Swift枚舉用例模式
Swift類型轉換模式
Swift表達式模式
Swift泛型參數
Swift泛型形參子句
Swift開發環境設置
Swift基本語法
Swift數據類型
Swift變量
Swift常量
Swift字面量
Swift運算符
Swift比較運算符
Swift邏輯運算符
Swift位運算符
Swift賦值運算符
Swift範圍運算符
Swift其它運算符
Swift運算符優先級
Swift算術運算符
Swift if語句
Swift if...else語句
Swift if...else if...else語句
Swift嵌套 if 語句
Swift Switch語句
Swift決策
Swift for-in循環
Swift for循環
Swift while循環
Swift do...while循環
Swift continue語句
Swift break語句
Swift fallthrough語句
Swift循環
Swift字符串
Swift字符
Swift數組
Swift函數
Swift閉包
Swift枚舉
Swift結構體
Swift類
Swift 屬性
Swift 方法
Swift 下標
Swift 繼承
Swift初始化
Swift 反初始化
Swift ARC自動引用計數
Swift 可選鏈
Swift 類型轉換
Swift 擴展
Swift 協議
Swift 泛型
Swift訪問控制

Swift解決實例之間的循環強引用

解決實例之間的循環強引用

Swift 提供了兩種辦法用來解決你在使用類的屬性時所遇到的循環強引用問題:弱引用(weak reference)和無主引用(unowned reference)。

弱引用和無主引用允許循環引用中的一個實例引用另外一個實例而不保持強引用。這樣實例能夠互相引用而不產生循環強引用。

對於生命週期中會變爲nil的實例使用弱引用。相反的,對於初始化賦值後再也不會被賦值爲nil的實例,使用無主引用。

弱引用

弱引用不會牢牢保持住引用的實例,並且不會阻止 ARC 銷燬被引用的實例。這種行爲阻止了引用變爲循環強引用。聲明屬性或者變量時,在前面加上weak關鍵字表明這是一個弱引用。

在實例的生命週期中,如果某些時候引用沒有值,那麼弱引用可以阻止循環強引用。如果引用總是有值,則可以使用無主引用,在無主引用中有描述。在上面Apartment的例子中,一個公寓的生命週期中,有時是沒有「居民」的,因此適合使用弱引用來解決循環強引用。

注意:
弱引用必須被聲明爲變量,表明其值能在運行時被修改。弱引用不能被聲明爲常量。  

因爲弱引用可以沒有值,你必須將每一個弱引用聲明爲可選類型。可選類型是在 Swift 語言中推薦的用來表示可能沒有值的類型。

因爲弱引用不會保持所引用的實例,即使引用存在,實例也有可能被銷燬。因此,ARC 會在引用的實例被銷燬後自動將其賦值爲nil。你可以像其他可選值一樣,檢查弱引用的值是否存在,你永遠也不會遇到被銷燬了而不存在的實例。

下面的例子跟上面PersonApartment的例子一致,但是有一個重要的區別。這一次,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") }
}

然後跟之前一樣,建立兩個變量(john和number73)之間的強引用,並關聯兩個實例:

var john: Person?
var number73: Apartment?

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

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

現在,兩個關聯在一起的實例的引用關係如下圖所示:

Swift解決實例之間的循環強引用

Person實例依然保持對Apartment實例的強引用,但是Apartment實例只是對Person實例的弱引用。這意味着當你斷開john變量所保持的強引用時,再也沒有指向Person實例的強引用了:

Swift解決實例之間的循環強引用

由於再也沒有指向Person實例的強引用,該實例會被銷燬:

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

唯一剩下的指向Apartment實例的強引用來自於變量number73。如果你斷開這個強引用,再也沒有指向Apartment實例的強引用了:

Swift解決實例之間的循環強引用

由於再也沒有指向Apartment實例的強引用,該實例也會被銷燬:

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

上面的兩段代碼展示了變量johnnumber73在被賦值爲nil後,Person實例和Apartment實例的析構函數都打印出「銷燬」的信息。這證明了引用循環被打破了。

無主引用

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

由於無主引用是非可選類型,你不需要在使用它的時候將它展開。無主引用總是可以被直接訪問。不過 ARC 無法在實例被銷燬後將無主引用設爲nil,因爲非可選類型的變量不允許被賦值爲nil

注意:
如果你試圖在實例被銷燬後,訪問該實例的無主引用,會觸發運行時錯誤。使用無主引用,你必須確保引用始終指向一個未銷燬的實例。
還需要注意的是如果你試圖訪問實例已經被銷燬的無主引用,程序會直接崩潰,而不會發生無法預期的行爲。所以你應當避免這樣的事情發生。  

下面的例子定義了兩個類,CustomerCreditCard,模擬了銀行客戶和客戶的信用卡。這兩個類中,每一個都將另外一個類的實例作爲自身的屬性。這種關係會潛在的創造循環強引用。

CustomerCreditCard之間的關係與前面弱引用例子中ApartmentPerson的關係截然不同。在這個數據模型中,一個客戶可能有或者沒有信用卡,但是一張信用卡總是關聯着一個客戶。爲了表示這種關係,Customer類有一個可選類型的card屬性,但是CreditCard類有一個非可選類型的customer屬性。

此外,只能通過將一個number值和customer實例傳遞給CreditCard構造函數的方式來創建CreditCard實例。這樣可以確保當創建CreditCard實例時總是有一個customer實例與之關聯。

由於信用卡總是關聯着一個客戶,因此將customer屬性定義爲無主引用,用以避免循環強引用:

class Customer {
    let name: String
    var card: CreditCard?
    init(name: String) {
        self.name = name
    }
    deinit { println("\(name) is being deinitialized") }
}
class CreditCard {
    let number: Int
    unowned let customer: Customer
    init(number: Int, customer: Customer) {
        self.number = number
        self.customer = customer
    }
    deinit { println("Card #\(number) is being deinitialized") }
}

下面的代碼片段定義了一個叫john的可選類型Customer變量,用來保存某個特定客戶的引用。由於是可選類型,所以變量被初始化爲nil

var john: Customer?

現在你可以創建Customer類的實例,用它初始化CreditCard實例,並將新創建的CreditCard實例賦值爲客戶的card屬性。

john = Customer(name: "John Appleseed")
john!.card = CreditCard(number: 1234_5678_9012_3456, customer: john!)

在你關聯兩個實例後,它們的引用關係如下圖所示:

Swift解決實例之間的循環強引用

Customer實例持有對CreditCard實例的強引用,而CreditCard實例持有對Customer實例的無主引用。

由於customer的無主引用,當你斷開john變量持有的強引用時,再也沒有指向Customer實例的強引用了:

Swift解決實例之間的循環強引用

由於再也沒有指向Customer實例的強引用,該實例被銷燬了。其後,再也沒有指向CreditCard實例的強引用,該實例也隨之被銷燬了:

john = nil
// prints "John Appleseed is being deinitialized"
// prints "Card #1234567890123456 is being deinitialized"

最後的代碼展示了在john變量被設爲nilCustomer實例和CreditCard實例的構造函數都打印出了「銷燬」的信息。

無主引用以及隱式解析可選屬性

上面弱引用和無主引用的例子涵蓋了兩種常用的需要打破循環強引用的場景。

PersonApartment的例子展示了兩個屬性的值都允許爲nil,並會潛在的產生循環強引用。這種場景最適合用弱引用來解決。

CustomerCreditCard的例子展示了一個屬性的值允許爲nil,而另一個屬性的值不允許爲nil,並會潛在的產生循環強引用。這種場景最適合通過無主引用來解決。

然而,存在着第三種場景,在這種場景中,兩個屬性都必須有值,並且初始化完成後不能爲nil。在這種場景中,需要一個類使用無主屬性,而另外一個類使用隱式解析可選屬性。

這使兩個屬性在初始化完成後能被直接訪問(不需要可選展開),同時避免了循環引用。這一節將爲你展示如何建立這種關係。

下面的例子定義了兩個類,CountryCity,每個類將另外一個類的實例保存爲屬性。在這個模型中,每個國家必須有首都,而每一個城市必須屬於一個國家。爲了實現這種關係,Country類擁有一個capitalCity屬性,而City類有一個country屬性:

class Country {
    let name: String
    let capitalCity: City!
    init(name: String, capitalName: String) {
        self.name = name
        self.capitalCity = City(name: capitalName, country: self)
    }
}
class City {
    let name: String
    unowned let country: Country
    init(name: String, country: Country) {
        self.name = name
        self.country = country
    }
}

爲了建立兩個類的依賴關係,City的構造函數有一個Country實例的參數,並且將實例保存爲country屬性。

Country的構造函數調用了City的構造函數。然而,只有Country的實例完全初始化完後,Country的構造函數才能把self傳給City的構造函數。(在兩段式構造過程中有具體描述)

爲了滿足這種需求,通過在類型結尾處加上感嘆號(City!)的方式,將CountrycapitalCity屬性聲明爲隱式解析可選類型的屬性。這表示像其他可選類型一樣,capitalCity屬性的默認值爲nil,但是不需要展開它的值就能訪問它。(在隱式解析可選類型中有描述)

由於capitalCity默認值爲nil,一旦Country的實例在構造函數中給name屬性賦值後,整個初始化過程就完成了。這代表一旦name屬性被賦值後,Country的構造函數就能引用並傳遞隱式的selfCountry的構造函數在賦值capitalCity時,就能將self作爲參數傳遞給City的構造函數。

以上的意義在於你可以通過一條語句同時創建CountryCity的實例,而不產生循環強引用,並且capitalCity的屬性能被直接訪問,而不需要通過感嘆號來展開它的可選值:

var country = Country(name: "Canada", capitalName: "Ottawa")
println("\(country.name)'s capital city is called \(country.capitalCity.name)")
// prints "Canada's capital city is called Ottawa"

在上面的例子中,使用隱式解析可選值的意義在於滿足了兩個類構造函數的需求。capitalCity屬性在初始化完成後,能像非可選值一樣使用和存取同時還避免了循環強引用。