自動引用計數

优质
小牛编辑
125浏览
2023-12-01

Swift 使用自動引用計數(ARC)這一機制來跟蹤和管理你的應用程式的內存。通常情況下,Swift 的內存管理機制會一直起著作用,你無須自己來考慮內存的管理。ARC 會在類別的實例不再被使用時,自動釋放其占用的內存。

然而,在少數情況下,ARC 為了能幫助你管理內存,需要更多的關於你的程式碼之間關系的資訊。本章描述了這些情況,並且為你示範怎樣啟用 ARC 來管理你的應用程式的內存。

注意:
參考計數僅僅應用於類別的實例。結構和列舉型別是值型別,不是參考型別,也不是通過參考的方式儲存和傳遞。

自動引用計數的工作機制

當你每次創建一個類別的新的實例的時候,ARC 會分配一大塊內存用來儲存實例的資訊。內存中會包含實例的型別資訊,以及這個實例所有相關屬性的值。此外,當實例不再被使用時,ARC 釋放實例所占用的內存,並讓釋放的內存能挪作他用。這確保了不再被使用的實例,不會一直占用內存空間。

然而,當 ARC 收回和釋放了正在被使用中的實例,該實例的屬性和方法將不能再被存取和呼叫。實際上,如果你試圖存取這個實例,你的應用程式很可能會崩潰。

為了確保使用中的實例不會被銷毀,ARC 會跟蹤和計算每一個實例正在被多少屬性,常數和變數所參考。哪怕實例的參考數為一,ARC都不會銷毀這個實例。

為了使之成為可能,無論你將實例賦值給屬性,常數或者是變數,屬性,常數或者變數,都會對此實例創建強參考。之所以稱之為強參考,是因為它會將實例牢牢的保持住,只要強參考還在,實例是不允許被銷毀的。

自動引用計數實踐

下面的範例展示了自動引用計數的工作機制。範例以一個簡單的Person類別開始,並定義了一個叫name的常數屬性:

class Person {
    let name: String
    init(name: String) {
        self.name = name
        println("\(name) is being initialized")
    }
    deinit {
        println("\(name) is being deinitialized")
    }
}

Person類別有一個建構函式,此建構函式為實例的name屬性賦值並列印出資訊,以表明初始化過程生效。Person類別同時也擁有析構函式,同樣會在實例被銷毀的時候列印出資訊。

接下來的程式碼片段定義了三個型別為Person?的變數,用來按照程式碼片段中的順序,為新的Person實例建立多個參考。由於這些變數是被定義為可選型別(Person?,而不是Person),它們的值會被自動初始化為nil,目前還不會參考到Person類別的實例。

var reference1: Person?
var reference2: Person?
var reference3: Person?

現在你可以創建Person類別的新實例,並且將它賦值給三個變數其中的一個:

reference1 = Person(name: "John Appleseed")
// prints "John Appleseed is being initialized」

應當注意到當你呼叫Person類別的建構函式的時候,"John Appleseed is being initialized」會被列印出來。由此可以確定建構函式被執行。

由於Person類別的新實例被賦值給了reference1變數,所以reference1Person類別的新實例之間建立了一個強參考。正是因為這個強參考,ARC 會保證Person實例被保持在內存中不被銷毀。

如果你將同樣的Person實例也賦值給其他兩個變數,該實例又會多出兩個強參考:

reference2 = reference1
reference3 = reference1

現在這個Person實例已經有三個強參考了。

如果你通過給兩個變數賦值nil的方式斷開兩個強參考()包括最先的那個強參考),只留下一個強參考,Person實例不會被銷毀:

reference2 = nil
reference3 = nil

ARC 會在第三個,也即最後一個強參考被斷開的時候,銷毀Person實例,這也意味著你不再使用這個Person實例:

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

類別實例之間的迴圈強參考

在上面的範例中,ARC 會跟蹤你所新創建的Person實例的參考數量,並且會在Person實例不再被需要時銷毀它。

然而,我們可能會寫出這樣的程式碼,一個類別永遠不會有0個強參考。這種情況發生在兩個類別實例互相保持對方的強參考,並讓對方不被銷毀。這就是所謂的迴圈強參考。

你可以通過定義類別之間的關系為弱參考或者無主參考,以此替代強參考,從而解決迴圈強參考的問題。具體的過程在解決類別實例之間的迴圈強參考中有描述。不管怎樣,在你學習怎樣解決迴圈強參考之前,很有必要了解一下它是怎樣產生的。

下面展示了一個不經意產生迴圈強參考的範例。範例定義了兩個類別:PersonApartment,用來建模公寓和它其中的居民:

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 }
    var tenant: Person?
    deinit { println("Apartment #\(number) is being deinitialized") }
}

每一個Person實例有一個型別為String,名字為name的屬性,並有一個可選的初始化為nilapartment屬性。apartment屬性是可選的,因為一個人並不總是擁有公寓。

類似的,每個Apartment實例有一個叫number,型別為Int的屬性,並有一個可選的初始化為niltenant屬性。tenant屬性是可選的,因為一棟公寓並不總是有居民。

這兩個類別都定義了析構函式,用以在類別實例被析構的時候輸出資訊。這讓你能夠知曉PersonApartment的實例是否像預期的那樣被銷毀。

接下來的程式碼片段定義了兩個可選型別的變數johnnumber73,並分別被設定為下面的ApartmentPerson的實例。這兩個變數都被初始化為nil,並為可選的:

var john: Person?
var number73: Apartment?

現在你可以創建特定的PersonApartment實例並將類別實例賦值給johnnumber73變數:

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

在兩個實例被創建和賦值後,下圖表現了強參考的關系。變數john現在有一個指向Person實例的強參考,而變數number73有一個指向Apartment實例的強參考:

現在你能夠將這兩個實例關聯在一起,這樣人就能有公寓住了,而公寓也有了房客。注意感嘆號是用來展開和存取可選變數johnnumber73中的實例,這樣實例的屬性才能被賦值:

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

在將兩個實例聯系在一起之後,強參考的關系如圖所示:

不幸的是,將這兩個實例關聯在一起之後,一個迴圈強參考被創建了。Person實例現在有了一個指向Apartment實例的強參考,而Apartment實例也有了一個指向Person實例的強參考。因此,當你斷開johnnumber73變數所持有的強參考時,參考計數並不會降為 0,實例也不會被 ARC 銷毀:

john = nil
number73 = nil

注意,當你把這兩個變數設為nil時,沒有任何一個析構函式被呼叫。強參考迴圈阻止了PersonApartment類別實例的銷毀,並在你的應用程式中造成了內存泄漏。

在你將johnnumber73賦值為nil後,強參考關系如下圖:

PersonApartment實例之間的強參考關系保留了下來並且不會被斷開。

解決實例之間的迴圈強參考

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

現在,兩個關聯在一起的實例的參考關系如下圖所示:

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

由於再也沒有指向Person實例的強參考,該實例會被銷毀:

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

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

由於再也沒有指向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!)

在你關聯兩個實例後,它們的參考關系如下圖所示:

Customer實例持有對CreditCard實例的強參考,而CreditCard實例持有對Customer實例的無主參考。

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

由於再也沒有指向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屬性在初始化完成後,能像非可選值一樣使用和存取同時還避免了迴圈強參考。

閉包引起的迴圈強參考

前面我們看到了迴圈強參考環是在兩個類別實例屬性互相保持對方的強參考時產生的,還知道了如何用弱參考和無主參考來打破迴圈強參考。

迴圈強參考還會發生在當你將一個閉包賦值給類別實例的某個屬性,並且這個閉包體中又使用了實例。這個閉包體中可能存取了實例的某個屬性,例如self.someProperty,或者閉包中呼叫了實例的某個方法,例如self.someMethod。這兩種情況都導致了閉包 「捕獲" self,從而產生了迴圈強參考。

迴圈強參考的產生,是因為閉包和類別相似,都是參考型別。當你把一個閉包賦值給某個屬性時,你也把一個參考賦值給了這個閉包。實質上,這跟之前的問題是一樣的-兩個強參考讓彼此一直有效。但是,和兩個類別實例不同,這次一個是類別實例,另一個是閉包。

Swift 提供了一種優雅的方法來解決這個問題,稱之為閉包占用列表(closuer capture list)。同樣的,在學習如何用閉包占用列表破壞迴圈強參考之前,先來了解一下迴圈強參考是如何產生的,這對我們是很有幫助的。

下面的範例為你展示了當一個閉包參考了self後是如何產生一個迴圈強參考的。範例中定義了一個叫HTMLElement的類別,用一種簡單的模型表示 HTML 中的一個單獨的元素:

class HTMLElement {

    let name: String
    let text: String?

    @lazy var asHTML: () -> String = {
        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類別定義了一個name屬性來表示這個元素的名稱,例如代表段落的"p",或者代表換行的"br"。HTMLElement還定義了一個可選屬性text,用來設置和展現 HTML 元素的文字。

除了上面的兩個屬性,HTMLElement還定義了一個lazy屬性asHTML。這個屬性參考了一個閉包,將nametext組合成 HTML 字串片段。該屬性是() -> String型別,或者可以理解為「一個沒有參數,回傳String的函式」。

預設情況下,閉包賦值給了asHTML屬性,這個閉包回傳一個代表 HTML 標簽的字串。如果text值存在,該標簽就包含可選值text;如果text不存在,該標簽就不包含文字。對於段落元素,根據text是"some text"還是nil,閉包會回傳"<p>some text</p>"或者"<p />"。

可以像實例方法那樣去命名、使用asHTML屬性。然而,由於asHTML是閉包而不是實例方法,如果你想改變特定元素的 HTML 處理的話,可以用自定義的閉包來取代預設值。

注意:
asHTML宣告為lazy屬性,因為只有當元素確實需要處理為HTML輸出的字串時,才需要使用asHTML。也就是說,在預設的閉包中可以使用self,因為只有當初始化完成以及self確實存在後,才能存取lazy屬性。

HTMLElement類別只提供一個建構函式,通過nametext(如果有的話)參數來初始化一個元素。該類別也定義了一個析構函式,當HTMLElement實例被銷毀時,列印一條訊息。

下面的程式碼展示了如何用HTMLElement類別創建實例並列印訊息。

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

注意:
上面的paragraph變數定義為可選HTMLElement,因此我們可以賦值nil給它來演示迴圈強參考。

不幸的是,上面寫的HTMLElement類別產生了類別實例和asHTML預設值的閉包之間的迴圈強參考。迴圈強參考如下圖所示:

實例的asHTML屬性持有閉包的強參考。但是,閉包在其閉包體內使用了self(參考了self.nameself.text),因此閉包捕獲了self,這意味著閉包又反過來持有了HTMLElement實例的強參考。這樣兩個物件就產生了迴圈強參考。(更多關於閉包捕獲值的資訊,請參考值捕獲)。

注意:
雖然閉包多次使用了self,它只捕獲HTMLElement實例的一個強參考。

如果設置paragraph變數為nil,打破它持有的HTMLElement實例的強參考,HTMLElement實例和它的閉包都不會被銷毀,也是因為迴圈強參考:

paragraph = nil

注意HTMLElementdeinitializer中的訊息並沒有別列印,證明了HTMLElement實例並沒有被銷毀。

解決閉包引起的迴圈強參考

在定義閉包時同時定義捕獲列表作為閉包的一部分,通過這種方式可以解決閉包和類別實例之間的迴圈強參考。捕獲列表定義了閉包體內捕獲一個或者多個參考型別的規則。跟解決兩個類別實例間的迴圈強參考一樣,宣告每個捕獲的參考為弱參考或無主參考,而不是強參考。應當根據程式碼關系來決定使用弱參考還是無主參考。

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

定義捕獲列表

捕獲列表中的每個元素都是由weak或者unowned關鍵字和實例的參考(如selfsomeInstance)成對組成。每一對都在方括號中,通過逗號分開。

捕獲列表放置在閉包參數列表和回傳型別之前:

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

如果閉包沒有指定參數列表或者回傳型別,則可以通過上下文推斷,那麼可以捕獲列表放在閉包開始的地方,跟著是關鍵字in

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

弱參考和無主參考

當閉包和捕獲的實例總是互相參考時並且總是同時銷毀時,將閉包內的捕獲定義為無主參考。

相反的,當捕獲參考有時可能會是nil時,將閉包內的捕獲定義為弱參考。弱參考總是可選型別,並且當參考的實例被銷毀後,弱參考的值會自動置為nil。這使我們可以在閉包內檢查它們是否存在。

注意:
如果捕獲的參考絕對不會置為nil,應該用無主參考,而不是弱參考。

前面的HTMLElement範例中,無主參考是正確的解決迴圈強參考的方法。這樣編寫HTMLElement類別來避免迴圈強參考:

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實作和之前的實作一致,只是在asHTML閉包中多了一個捕獲列表。這裡,捕獲列表是[unowned self],表示「用無主參考而不是強參考來捕獲self」。

和之前一樣,我們可以創建並列印HTMLElement實例:

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

使用捕獲列表後參考關系如下圖所示:

這一次,閉包以無主參考的形式捕獲self,並不會持有HTMLElement實例的強參考。如果將paragraph賦值為nilHTMLElement實例將會被銷毀,並能看到它的析構函式列印出的訊息。

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