# 自動引用計數
Swift 使用*自動引用計數(ARC)*機制來跟蹤和管理你的應用程序的內存。通常情況下,Swift 內存管理機制會一直起作用,你無須自己來考慮內存的管理。ARC 會在類的實例不再被使用時,自動釋放其占用的內存。
然而在少數情況下,為了能幫助你管理內存,ARC 需要更多的,代碼之間關系的信息。本章描述了這些情況,并且為你示范怎樣才能使 ARC 來管理你的應用程序的所有內存。在 Swift 使用 ARC 與在 Obejctive-C 中使用 ARC 非常類似,具體請參考 [過渡到 ARC 的發布說明](https://developer.apple.com/library/content/releasenotes/ObjectiveC/RN-TransitioningToARC/Introduction/Introduction.html#//apple-ref/doc/uid/TP40011226)。
> 注意
>
> 引用計數僅僅應用于類的實例。結構體和枚舉類型是值類型,不是引用類型,也不是通過引用的方式存儲和傳遞。
## 自動引用計數的工作機制 {#how-arc-works}
每當你創建一個新的類實例時,ARC 會分配一塊內存來儲存該實例的信息。內存中會包含實例的類型信息,以及這個實例所關聯的任何存儲屬性的值。
此外,當實例不再被使用時,ARC 釋放實例所占用的內存,并讓釋放的內存能挪作他用。這確保了不再被使用的實例,不會一直占用內存空間。
然而,當 ARC 回收并釋放了正在被使用中的實例后,該實例的屬性和方法將不能再被訪問和調用。實際上,如果你試圖訪問這個實例,你的應用程序很可能會崩潰。
為了確保使用中的實例不會被銷毀,ARC 會跟蹤和計算每一個實例正在被多少屬性,常量和變量所引用。哪怕實例的引用數為 1,ARC 都不會銷毀這個實例。
為了使上述成為可能,無論你將實例賦值給屬性、常量或變量,它們都會創建此實例的強引用。之所以稱之為“強”引用,是因為它會將實例牢牢地保持住,只要強引用還在,實例是不允許被銷毀的。
## 自動引用計數實踐 {#arc-in-action}
下面的例子展示了自動引用計數的工作機制。例子以一個簡單的 `Person` 類開始,并定義了一個叫 `name` 的常量屬性:
```swift
class Person {
let name: String
init(name: String) {
self.name = name
print("\(name) is being initialized")
}
deinit {
print("\(name) is being deinitialized")
}
}
```
`Person` 類有一個構造器,此構造器給實例的 `name` 屬性賦值,并打印一條消息以表明初始化過程生效。`Person` 類也擁有一個析構器,這個析構器會在實例被銷毀時打印一條消息。
接下來的代碼片段定義了三個類型為 `Person?` 的變量,按照代碼片段中的順序,為新的 `Person` 實例建立多個引用。由于這些變量是被定義為可選類型(`Person?`,而不是 `Person`),它們的值會被自動初始化為 `nil`,目前還不會引用到 `Person` 類的實例。
```swift
var reference1: Person?
var reference2: Person?
var reference3: Person?
```
現在你可以創建 `Person` 類的新實例,并且將它賦值給三個變量中的一個:
```swift
reference1 = Person(name: "John Appleseed")
// 打印“John Appleseed is being initialized”
```
應當注意到當你調用 `Person` 類的構造器的時候,`"John Appleseed is being initialized"` 會被打印出來。由此可以確定構造器被執行。
由于 `Person` 類的新實例被賦值給了 `reference1` 變量,所以 `reference1` 到 `Person` 類的新實例之間建立了一個強引用。正是因為這一個強引用,ARC 會保證 `Person` 實例被保持在內存中不被銷毀。
如果你將同一個 `Person` 實例也賦值給其他兩個變量,該實例又會多出兩個強引用:
```swift
reference2 = reference1
reference3 = reference1
```
現在這一個 `Person` 實例已經有三個強引用了。
如果你通過給其中兩個變量賦值 `nil` 的方式斷開兩個強引用(包括最先的那個強引用),只留下一個強引用,`Person` 實例不會被銷毀:
```swift
reference1 = nil
reference2 = nil
```
在你清楚地表明不再使用這個 `Person` 實例時,即第三個也就是最后一個強引用被斷開時,ARC 會銷毀它:
```swift
reference3 = nil
// 打印“John Appleseed is being deinitialized”
```
## 類實例之間的循環強引用 {#strong-reference-cycles-between-class-instances}
在上面的例子中,ARC 會跟蹤你所新創建的 `Person` 實例的引用數量,并且會在 `Person` 實例不再被需要時銷毀它。
然而,我們可能會寫出一個類實例的強引用數*永遠不能*變成 `0` 的代碼。如果兩個類實例互相持有對方的強引用,因而每個實例都讓對方一直存在,就是這種情況。這就是所謂的*循環強引用*。
你可以通過定義類之間的關系為弱引用或無主引用,來替代強引用,從而解決循環強引用的問題。具體的過程在 [解決類實例之間的循環強引用](#resolving-strong-reference-cycles-between-class-instances) 中有描述。不管怎樣,在你學習怎樣解決循環強引用之前,很有必要了解一下它是怎樣產生的。
下面展示了一個不經意產生循環強引用的例子。例子定義了兩個類:`Person` 和 `Apartment`,用來建模公寓和它其中的居民:
```swift
class Person {
let name: String
init(name: String) { self.name = name }
var apartment: Apartment?
deinit { print("\(name) is being deinitialized") }
}
class Apartment {
let unit: String
init(unit: String) { self.unit = unit }
var tenant: Person?
deinit { print("Apartment \(unit) is being deinitialized") }
}
```
每一個 `Person` 實例有一個類型為 `String`,名字為 `name` 的屬性,并有一個可選的初始化為 `nil` 的 `apartment` 屬性。`apartment` 屬性是可選的,因為一個人并不總是擁有公寓。
類似的,每個 `Apartment` 實例有一個叫 `unit`,類型為 `String` 的屬性,并有一個可選的初始化為 `nil` 的 `tenant` 屬性。`tenant` 屬性是可選的,因為一棟公寓并不總是有居民。
這兩個類都定義了析構器,在類實例被析構的時候輸出信息。這讓你能夠知曉 `Person` 和 `Apartment` 的實例是否像預期的那樣被銷毀。
接下來的代碼片段定義了兩個可選類型的變量 `john` 和 `unit4A`,并分別被設為下面的 `Apartment` 和 `Person` 的實例。這兩個變量都被初始化為 `nil`,這正是可選類型的優點:
```swift
var john: Person?
var unit4A: Apartment?
```
現在你可以創建特定的 `Person` 和 `Apartment` 實例并將賦值給 `john` 和 `unit4A` 變量:
```swift
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
```
在兩個實例被創建和賦值后,下圖表現了強引用的關系。變量 `john` 現在有一個指向 `Person` 實例的強引用,而變量 `unit4A` 有一個指向 `Apartment` 實例的強引用:

現在你能夠將這兩個實例關聯在一起,這樣人就能有公寓住了,而公寓也有了房客。注意感嘆號是用來解包和訪問可選變量 `john` 和 `unit4A` 中的實例,這樣實例的屬性才能被賦值:
```swift
john!.apartment = unit4A
unit4A!.tenant = john
```
在將兩個實例聯系在一起之后,強引用的關系如圖所示:

不幸的是,這兩個實例關聯后會產生一個循環強引用。`Person` 實例現在有了一個指向 `Apartment` 實例的強引用,而 `Apartment` 實例也有了一個指向 `Person` 實例的強引用。因此,當你斷開 `john` 和 `unit4A` 變量所持有的強引用時,引用計數并不會降為 `0`,實例也不會被 ARC 銷毀:
```swift
john = nil
unit4A = nil
```
注意,當你把這兩個變量設為 `nil` 時,沒有任何一個析構器被調用。循環強引用會一直阻止 `Person` 和 `Apartment` 類實例的銷毀,這就在你的應用程序中造成了內存泄漏。
在你將 `john` 和 `unit4A` 賦值為 `nil` 后,強引用關系如下圖:

`Person` 和 `Apartment` 實例之間的強引用關系保留了下來并且不會被斷開。
## 解決實例之間的循環強引用 {#resolving-strong-reference-cycles-between-class-instances}
Swift 提供了兩種辦法用來解決你在使用類的屬性時所遇到的循環強引用問題:弱引用(weak reference)和無主引用(unowned reference)。
弱引用和無主引用允許循環引用中的一個實例引用另一個實例而*不*保持強引用。這樣實例能夠互相引用而不產生循環強引用。
當其他的實例有更短的生命周期時,使用弱引用,也就是說,當其他實例析構在先時。在上面公寓的例子中,很顯然一個公寓在它的生命周期內會在某個時間段沒有它的主人,所以一個弱引用就加在公寓類里面,避免循環引用。相比之下,當其他實例有相同的或者更長生命周期時,請使用無主引用。
### 弱引用 {#weak-references}
*弱引用*不會對其引用的實例保持強引用,因而不會阻止 ARC 銷毀被引用的實例。這個特性阻止了引用變為循環強引用。聲明屬性或者變量時,在前面加上 `weak` 關鍵字表明這是一個弱引用。
因為弱引用不會保持所引用的實例,即使引用存在,實例也有可能被銷毀。因此,ARC 會在引用的實例被銷毀后自動將其弱引用賦值為 `nil`。并且因為弱引用需要在運行時允許被賦值為 `nil`,所以它們會被定義為可選類型變量,而不是常量。
你可以像其他可選值一樣,檢查弱引用的值是否存在,這樣可以避免訪問已銷毀的實例的引用。
> 注意
>
> 當 ARC 設置弱引用為 `nil` 時,屬性觀察不會被觸發。
下面的例子跟上面 `Person` 和 `Apartment` 的例子一致,但是有一個重要的區別。這一次,`Apartment` 的 `tenant` 屬性被聲明為弱引用:
```swift
class Person {
let name: String
init(name: String) { self.name = name }
var apartment: Apartment?
deinit { print("\(name) is being deinitialized") }
}
class Apartment {
let unit: String
init(unit: String) { self.unit = unit }
weak var tenant: Person?
deinit { print("Apartment \(unit) is being deinitialized") }
}
```
然后跟之前一樣,建立兩個變量(`john` 和 `unit4A`)之間的強引用,并關聯兩個實例:
```swift
var john: Person?
var unit4A: Apartment?
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
john!.apartment = unit4A
unit4A!.tenant = john
```
現在,兩個關聯在一起的實例的引用關系如下圖所示:

`Person` 實例依然保持對 `Apartment` 實例的強引用,但是 `Apartment` 實例只持有對 `Person` 實例的弱引用。這意味著當你通過把 `john` 變量賦值為 `nil` 而斷開其所保持的強引用時,再也沒有指向 `Person` 實例的強引用了:
```swift
john = nil
// 打印“John Appleseed is being deinitialized”
```
由于再也沒有指向 `Person` 實例的強引用,該實例會被銷毀,且 `tenant` 屬性會被賦值為 `nil`:

唯一剩下的指向 `Apartment` 實例的強引用來自于變量 `unit4A`。如果你斷開這個強引用,再也沒有指向 `Apartment` 實例的強引用了:
```swift
unit4A = nil
// 打印“Apartment 4A is being deinitialized”
```
由于再也沒有指向 `Person` 實例的強引用,該實例會被銷毀:

> 注意
>
> 在使用垃圾收集的系統里,弱指針有時用來實現簡單的緩沖機制,因為沒有強引用的對象只會在內存壓力觸發垃圾收集時才被銷毀。但是在 ARC 中,一旦值的最后一個強引用被移除,就會被立即銷毀,這導致弱引用并不適合上面的用途。
### 無主引用 {#unowned-references}
和弱引用類似,*無主引用*不會牢牢保持住引用的實例。和弱引用不同的是,無主引用在其他實例有相同或者更長的生命周期時使用。你可以在聲明屬性或者變量時,在前面加上關鍵字 `unowned` 表示這是一個無主引用。
無主引用通常都被期望擁有值。不過 ARC 無法在實例被銷毀后將無主引用設為 `nil`,因為非可選類型的變量不允許被賦值為 `nil`。
> 重點
>
> 使用無主引用,你*必須*確保引用始終指向一個未銷毀的實例。
>
> 如果你試圖在實例被銷毀后,訪問該實例的無主引用,會觸發運行時錯誤。
下面的例子定義了兩個類,`Customer` 和 `CreditCard`,模擬了銀行客戶和客戶的信用卡。這兩個類中,每一個都將另外一個類的實例作為自身的屬性。這種關系可能會造成循環強引用。
`Customer` 和 `CreditCard` 之間的關系與前面弱引用例子中 `Apartment` 和 `Person` 的關系略微不同。在這個數據模型中,一個客戶可能有或者沒有信用卡,但是一張信用卡總是關聯著一個客戶。為了表示這種關系,`Customer` 類有一個可選類型的 `card` 屬性,但是 `CreditCard` 類有一個非可選類型的 `customer` 屬性。
此外,只能通過將一個 `number` 值和 `customer` 實例傳遞給 `CreditCard` 構造器的方式來創建 `CreditCard` 實例。這樣可以確保當創建 `CreditCard` 實例時總是有一個 `customer` 實例與之關聯。
由于信用卡總是關聯著一個客戶,因此將 `customer` 屬性定義為無主引用,用以避免循環強引用:
```swift
class Customer {
let name: String
var card: CreditCard?
init(name: String) {
self.name = name
}
deinit { print("\(name) is being deinitialized") }
}
class CreditCard {
let number: UInt64
unowned let customer: Customer
init(number: UInt64, customer: Customer) {
self.number = number
self.customer = customer
}
deinit { print("Card #\(number) is being deinitialized") }
}
```
> 注意
>
> `CreditCard` 類的 `number` 屬性被定義為 `UInt64` 類型而不是 `Int` 類型,以確保 `number` 屬性的存儲量在 32 位和 64 位系統上都能足夠容納 16 位的卡號。
下面的代碼片段定義了一個叫 `john` 的可選類型 `Customer` 變量,用來保存某個特定客戶的引用。由于是可選類型,所以變量被初始化為 `nil`:
```swift
var john: Customer?
```
現在你可以創建 `Customer` 類的實例,用它初始化 `CreditCard` 實例,并將新創建的 `CreditCard` 實例賦值為客戶的 `card` 屬性:
```swift
john = Customer(name: "John Appleseed")
john!.card = CreditCard(number: 1234_5678_9012_3456, customer: john!)
```
在你關聯兩個實例后,它們的引用關系如下圖所示:

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

由于再也沒有指向 `Customer` 實例的強引用,該實例被銷毀了。其后,再也沒有指向 `CreditCard` 實例的強引用,該實例也隨之被銷毀了:
```swift
john = nil
// 打印“John Appleseed is being deinitialized”
// 打印“Card #1234567890123456 is being deinitialized”
```
最后的代碼展示了在 `john` 變量被設為 `nil` 后 `Customer` 實例和 `CreditCard` 實例的析構器都打印出了“銷毀”的信息。
> 注意
>
> 上面的例子展示了如何使用安全的無主引用。對于需要禁用運行時的安全檢查的情況(例如,出于性能方面的原因),Swift 還提供了不安全的無主引用。與所有不安全的操作一樣,你需要負責檢查代碼以確保其安全性。
> 你可以通過 `unowned(unsafe)` 來聲明不安全無主引用。如果你試圖在實例被銷毀后,訪問該實例的不安全無主引用,你的程序會嘗試訪問該實例之前所在的內存地址,這是一個不安全的操作。
### 無主引用和隱式解包可選值屬性 {#unowned-references-and-implicitly-unwrapped-optional-properties}
上面弱引用和無主引用的例子涵蓋了兩種常用的需要打破循環強引用的場景。
`Person` 和 `Apartment` 的例子展示了兩個屬性的值都允許為 `nil`,并會潛在的產生循環強引用。這種場景最適合用弱引用來解決。
`Customer` 和 `CreditCard` 的例子展示了一個屬性的值允許為 `nil`,而另一個屬性的值不允許為 `nil`,這也可能會產生循環強引用。這種場景最適合通過無主引用來解決。
然而,存在著第三種場景,在這種場景中,兩個屬性都必須有值,并且初始化完成后永遠不會為 `nil`。在這種場景中,需要一個類使用無主屬性,而另外一個類使用隱式解包可選值屬性。
這使兩個屬性在初始化完成后能被直接訪問(不需要可選解包),同時避免了循環引用。這一節將為你展示如何建立這種關系。
下面的例子定義了兩個類,`Country` 和 `City`,每個類將另外一個類的實例保存為屬性。在這個模型中,每個國家必須有首都,每個城市必須屬于一個國家。為了實現這種關系,`Country` 類擁有一個 `capitalCity` 屬性,而 `City` 類有一個 `country` 屬性:
```swift
class Country {
let name: String
var 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` 的構造器。在 [兩段式構造過程](./14_Initialization.md#two-phase-initialization) 中有具體描述。
為了滿足這種需求,通過在類型結尾處加上感嘆號(`City!`)的方式,將 `Country` 的 `capitalCity` 屬性聲明為隱式解包可選值類型的屬性。這意味著像其他可選類型一樣,`capitalCity` 屬性的默認值為 `nil`,但是不需要解包它的值就能訪問它。在 [隱式解包可選值](./01_The_Basics.md#implicityly-unwrapped-optionals) 中有描述。
由于 `capitalCity` 默認值為 `nil`,一旦 `Country` 的實例在構造器中給 `name` 屬性賦值后,整個初始化過程就完成了。這意味著一旦 `name` 屬性被賦值后,`Country` 的構造器就能引用并傳遞隱式的 `self`。`Country` 的構造器在賦值 `capitalCity` 時,就能將 `self` 作為參數傳遞給 `City` 的構造器。
上述的意義在于你可以通過一條語句同時創建 `Country` 和 `City` 的實例,而不產生循環強引用,并且 `capitalCity` 的屬性能被直接訪問,而不需要通過感嘆號來解包它的可選值:
```swift
var country = Country(name: "Canada", capitalName: "Ottawa")
print("\(country.name)'s capital city is called \(country.capitalCity.name)")
// 打印“Canada's capital city is called Ottawa”
```
在上面的例子中,使用隱式解包可選值值意味著滿足了類的構造器的兩個構造階段的要求。`capitalCity` 屬性在初始化完成后,能像非可選值一樣使用和存取,同時還避免了循環強引用。
## 閉包的循環強引用 {#strong-reference-cycles-for-closures}
前面我們看到了循環強引用是在兩個類實例屬性互相保持對方的強引用時產生的,還知道了如何用弱引用和無主引用來打破這些循環強引用。
循環強引用還會發生在當你將一個閉包賦值給類實例的某個屬性,并且這個閉包體中又使用了這個類實例時。這個閉包體中可能訪問了實例的某個屬性,例如 `self.someProperty`,或者閉包中調用了實例的某個方法,例如 `self.someMethod()`。這兩種情況都導致了閉包“捕獲”`self`,從而產生了循環強引用。
循環強引用的產生,是因為閉包和類相似,都是引用類型。當你把一個閉包賦值給某個屬性時,你是將這個閉包的引用賦值給了屬性。實質上,這跟之前的問題是一樣的——兩個強引用讓彼此一直有效。但是,和兩個類實例不同,這次一個是類實例,另一個是閉包。
Swift 提供了一種優雅的方法來解決這個問題,稱之為 `閉包捕獲列表`(closure capture list)。同樣的,在學習如何用閉包捕獲列表打破循環強引用之前,先來了解一下這里的循環強引用是如何產生的,這對我們很有幫助。
下面的例子為你展示了當一個閉包引用了 `self` 后是如何產生一個循環強引用的。例子中定義了一個叫 `HTMLElement` 的類,用一種簡單的模型表示 HTML 文檔中的一個單獨的元素:
```swift
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 {
print("\(name) is being deinitialized")
}
}
```
`HTMLElement` 類定義了一個 `name` 屬性來表示這個元素的名稱,例如代表頭部元素的 `"h1"`,代表段落的 `"p"`,或者代表換行的 `"br"`。`HTMLElement` 還定義了一個可選屬性 `text`,用來設置 HTML 元素呈現的文本。
除了上面的兩個屬性,`HTMLElement` 還定義了一個 `lazy` 屬性 `asHTML`。這個屬性引用了一個將 `name` 和 `text` 組合成 HTML 字符串片段的閉包。該屬性是 `Void -> String` 類型,或者可以理解為“一個沒有參數,返回 `String` 的函數”。
默認情況下,閉包賦值給了 `asHTML` 屬性,這個閉包返回一個代表 HTML 標簽的字符串。如果 `text` 值存在,該標簽就包含可選值 `text`;如果 `text` 不存在,該標簽就不包含文本。對于段落元素,根據 `text` 是 `"some text"` 還是 `nil`,閉包會返回 `"<p>some text</p>"` 或者 `"<p />"`。
可以像實例方法那樣去命名、使用 `asHTML` 屬性。然而,由于 `asHTML` 是閉包而不是實例方法,如果你想改變特定 HTML 元素的處理方式的話,可以用自定義的閉包來取代默認值。
例如,可以將一個閉包賦值給 `asHTML` 屬性,這個閉包能在 `text` 屬性是 `nil` 時使用默認文本,這是為了避免返回一個空的 HTML 標簽:
```swift
let heading = HTMLElement(name: "h1")
let defaultText = "some default text"
heading.asHTML = {
return "<\(heading.name)>\(heading.text ?? defaultText)</\(heading.name)>"
}
print(heading.asHTML())
// 打印“<h1>some default text</h1>”
```
> 注意
>
> `asHTML` 聲明為 `lazy` 屬性,因為只有當元素確實需要被處理為 HTML 輸出的字符串時,才需要使用 `asHTML`。也就是說,在默認的閉包中可以使用 `self`,因為只有當初始化完成以及 `self` 確實存在后,才能訪問 `lazy` 屬性。
`HTMLElement` 類只提供了一個構造器,通過 `name` 和 `text`(如果有的話)參數來初始化一個新元素。該類也定義了一個析構器,當 `HTMLElement` 實例被銷毀時,打印一條消息。
下面的代碼展示了如何用 `HTMLElement` 類創建實例并打印消息:
```swift
var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
print(paragraph!.asHTML())
// 打印“<p>hello, world</p>”
```
> 注意
>
> 上面的 `paragraph` 變量定義為可選類型的 `HTMLElement`,因此我們可以賦值 `nil` 給它來演示循環強引用。
不幸的是,上面寫的 `HTMLElement` 類產生了類實例和作為 `asHTML` 默認值的閉包之間的循環強引用。循環強引用如下圖所示:

實例的 `asHTML` 屬性持有閉包的強引用。但是,閉包在其閉包體內使用了 `self`(引用了 `self.name` 和 `self.text`),因此閉包捕獲了 `self`,這意味著閉包又反過來持有了 `HTMLElement` 實例的強引用。這樣兩個對象就產生了循環強引用。(更多關于閉包捕獲值的信息,請參考 [值捕獲](./07_Closures.md#capturing-values))。
> 注意
>
> 雖然閉包多次使用了 `self`,它只捕獲 `HTMLElement` 實例的一個強引用。
如果設置 `paragraph` 變量為 `nil`,打破它持有的 `HTMLElement` 實例的強引用,`HTMLElement` 實例和它的閉包都不會被銷毀,也是因為循環強引用:
```swift
paragraph = nil
```
注意,`HTMLElement` 的析構器中的消息并沒有被打印,證明了 `HTMLElement` 實例并沒有被銷毀。
## 解決閉包的循環強引用 {#resolving-strong-reference-cycles-for-closures}
在定義閉包時同時定義捕獲列表作為閉包的一部分,通過這種方式可以解決閉包和類實例之間的循環強引用。捕獲列表定義了閉包體內捕獲一個或者多個引用類型的規則。跟解決兩個類實例間的循環強引用一樣,聲明每個捕獲的引用為弱引用或無主引用,而不是強引用。應當根據代碼關系來決定使用弱引用還是無主引用。
> 注意
>
> Swift 有如下要求:只要在閉包內使用 `self` 的成員,就要用 `self.someProperty` 或者 `self.someMethod()`(而不只是 `someProperty` 或 `someMethod()`)。這提醒你可能會一不小心就捕獲了 `self`。
### 定義捕獲列表 {#defining-a-capture-list}
捕獲列表中的每一項都由一對元素組成,一個元素是 `weak` 或 `unowned` 關鍵字,另一個元素是類實例的引用(例如 `self`)或初始化過的變量(如 `delegate = self.delegate`)。這些項在方括號中用逗號分開。
如果閉包有參數列表和返回類型,把捕獲列表放在它們前面:
```swift
lazy var someClosure = {
[unowned self, weak delegate = self.delegate]
(index: Int, stringToProcess: String) -> String in
// 這里是閉包的函數體
}
```
如果閉包沒有指明參數列表或者返回類型,它們會通過上下文推斷,那么可以把捕獲列表和關鍵字 `in` 放在閉包最開始的地方:
```swift
lazy var someClosure = {
[unowned self, weak delegate = self.delegate] in
// 這里是閉包的函數體
}
```
### 弱引用和無主引用 {#weak-and-unowned-references}
在閉包和捕獲的實例總是互相引用并且總是同時銷毀時,將閉包內的捕獲定義為 `無主引用`。
相反的,在被捕獲的引用可能會變為 `nil` 時,將閉包內的捕獲定義為 `弱引用`。弱引用總是可選類型,并且當引用的實例被銷毀后,弱引用的值會自動置為 `nil`。這使我們可以在閉包體內檢查它們是否存在。
> 注意
>
> 如果被捕獲的引用絕對不會變為 `nil`,應該用無主引用,而不是弱引用。
前面的 `HTMLElement` 例子中,無主引用是正確的解決循環強引用的方法。這樣編寫 `HTMLElement` 類來避免循環強引用:
```swift
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 {
print("\(name) is being deinitialized")
}
}
```
上面的 `HTMLElement` 實現和之前的實現一致,除了在 `asHTML` 閉包中多了一個捕獲列表。這里,捕獲列表是 `[unowned self]`,表示“將 `self` 捕獲為無主引用而不是強引用”。
和之前一樣,我們可以創建并打印 `HTMLElement` 實例:
```swift
var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
print(paragraph!.asHTML())
// 打印“<p>hello, world</p>”
```
使用捕獲列表后引用關系如下圖所示:

這一次,閉包以無主引用的形式捕獲 `self`,并不會持有 `HTMLElement` 實例的強引用。如果將 `paragraph` 賦值為 `nil`,`HTMLElement` 實例將會被銷毀,并能看到它的析構器打印出的消息:
```swift
paragraph = nil
// 打印“p is being deinitialized”
```
你可以查看 [捕獲列表](../03_language_reference/04_Expressions.md) 章節,獲取更多關于捕獲列表的信息。
- 1.關于 Swift
- 2.Swift 初見
- 2-1基礎部分
- 2-2基本運算符
- 2-3字符串和字符
- 2-4集合類型
- 2-5控制流
- 2-6函數
- 2-7閉包
- 2-8枚舉
- 2-9類和結構體
- 2-10屬性
- 2-11方法
- 2-12下標
- 2-13繼承
- 2-14構造過程
- 2-15析構過程
- 2-16可選鏈
- 2-17錯誤處理
- 2-18類型轉換
- 2-19嵌套類型
- 2-20擴展
- 2-21協議
- 2-22泛型
- 2-23不透明類型
- 2-24自動引用計數
- 2-25內存安全
- 2-26訪問控制
- 2-27高級運算符
- 3-1關于語言參考
- 3-2詞法結構
- 3-3類型
- 3-4表達式
- 3-5語句
- 3-6聲明
- 3-7特性
- 3-8模式
- 3-9泛型參數
- 4語法總結