Swift基礎(chǔ)-13(自動(dòng)引用計(jì)數(shù))

當(dāng)你每次創(chuàng)建一個(gè)類的新的實(shí)例的時(shí)候赊舶,ARC 會(huì)分配一塊內(nèi)存來(lái)儲(chǔ)存該實(shí)例信息淤刃。內(nèi)存中會(huì)包含實(shí)例的類型信息晒他,以及這個(gè)實(shí)例所有相關(guān)的存儲(chǔ)型屬性的值。

此外逸贾,當(dāng)實(shí)例不再被使用時(shí)陨仅,ARC 釋放實(shí)例所占用的內(nèi)存,并讓釋放的內(nèi)存能挪作他用铝侵。這確保了不再被使用的實(shí)例灼伤,不會(huì)一直占用內(nèi)存空間。

然而咪鲜,當(dāng) ARC 收回和釋放了正在被使用中的實(shí)例狐赡,該實(shí)例的屬性和方法將不能再被訪問(wèn)和調(diào)用。實(shí)際上疟丙,如果你試圖訪問(wèn)這個(gè)實(shí)例颖侄,你的應(yīng)用程序很可能會(huì)崩潰。

為了確保使用中的實(shí)例不會(huì)被銷毀享郊,ARC 會(huì)跟蹤和計(jì)算每一個(gè)實(shí)例正在被多少屬性览祖,常量和變量所引用。哪怕實(shí)例的引用數(shù)為1炊琉,ARC都不會(huì)銷毀這個(gè)實(shí)例展蒂。

為了使上述成為可能,無(wú)論你將實(shí)例賦值給屬性苔咪、常量或變量锰悼,它們都會(huì)創(chuàng)建此實(shí)例的強(qiáng)引用。之所以稱之為“強(qiáng)”引用悼泌,是因?yàn)樗鼤?huì)將實(shí)例牢牢地保持住松捉,只要強(qiáng)引用還在,實(shí)例是不允許被銷毀的馆里。

1.自動(dòng)引用計(jì)數(shù)實(shí)踐

下面的例子展示了自動(dòng)引用計(jì)數(shù)的工作機(jī)制隘世。例子以一個(gè)簡(jiǎn)單的Person類開始,并定義了一個(gè)叫name的常量屬性:

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

Person類有一個(gè)構(gòu)造函數(shù)鸠踪,此構(gòu)造函數(shù)為實(shí)例的name屬性賦值丙者,并打印一條消息以表明初始化過(guò)程生效。Person類也擁有一個(gè)析構(gòu)函數(shù)营密,這個(gè)析構(gòu)函數(shù)會(huì)在實(shí)例被銷毀時(shí)打印一條消息械媒。

接下來(lái)的代碼片段定義了三個(gè)類型為Person?的變量,用來(lái)按照代碼片段中的順序评汰,為新的Person實(shí)例建立多個(gè)引用纷捞。由于這些變量是被定義為可選類型(Person?,而不是Person)被去,它們的值會(huì)被自動(dòng)初始化為nil主儡,目前還不會(huì)引用到Person類的實(shí)例。

“var reference1: Person?
var reference2: Person?
var reference3: Person?”

“現(xiàn)在你可以創(chuàng)建Person類的新實(shí)例惨缆,并且將它賦值給三個(gè)變量中的一個(gè):”

“reference1 = Person(name: "John Appleseed")
// 打印 "John Appleseed is being initialized”

“應(yīng)當(dāng)注意到當(dāng)你調(diào)用Person類的構(gòu)造函數(shù)的時(shí)候糜值,“John Appleseed is being initialized”會(huì)被打印出來(lái)。由此可以確定構(gòu)造函數(shù)被執(zhí)行坯墨。

由于Person類的新實(shí)例被賦值給了reference1變量寂汇,所以reference1到Person類的新實(shí)例之間建立了一個(gè)強(qiáng)引用。正是因?yàn)檫@一個(gè)強(qiáng)引用捣染,ARC 會(huì)保證Person實(shí)例被保持在內(nèi)存中不被銷毀骄瓣。

如果你將同一個(gè)Person實(shí)例也賦值給其他兩個(gè)變量,該實(shí)例又會(huì)多出兩個(gè)強(qiáng)引用:”

reference2 = reference1
reference3 = reference1

“現(xiàn)在這一個(gè)Person實(shí)例已經(jīng)有三個(gè)強(qiáng)引用了耍攘。
如果你通過(guò)給其中兩個(gè)變量賦值nil的方式斷開兩個(gè)強(qiáng)引用(包括最先的那個(gè)強(qiáng)引用)累贤,只留下一個(gè)強(qiáng)引用,Person實(shí)例不會(huì)被銷毀:”

reference1 = nil
reference2 = nil

“在你清楚地表明不再使用這個(gè)Person實(shí)例時(shí)少漆,即第三個(gè)也就是最后一個(gè)強(qiáng)引用被斷開時(shí)臼膏,ARC 會(huì)銷毀它:”

reference3 = nil
// Prints "John Appleseed is being deinitialized"
2.類實(shí)例之間的循環(huán)強(qiáng)引用

“在上面的例子中,ARC 會(huì)跟蹤你所新創(chuàng)建的Person實(shí)例的引用數(shù)量示损,并且會(huì)在Person實(shí)例不再被需要時(shí)銷毀它渗磅。

然而,我們可能會(huì)寫出一個(gè)類實(shí)例的強(qiáng)引用數(shù)永遠(yuǎn)不能變成0的代碼检访。如果兩個(gè)類實(shí)例互相持有對(duì)方的強(qiáng)引用始鱼,因而每個(gè)實(shí)例都讓對(duì)方一直存在,就是這種情況脆贵。這就是所謂的循環(huán)強(qiáng)引用医清。

你可以通過(guò)定義類之間的關(guān)系為弱引用或無(wú)主引用,以替代強(qiáng)引用卖氨,從而解決循環(huán)強(qiáng)引用的問(wèn)題会烙。具體的過(guò)程在解決類實(shí)例之間的循環(huán)強(qiáng)引用中有描述负懦。不管怎樣,在你學(xué)習(xí)怎樣解決循環(huán)強(qiáng)引用之前柏腻,很有必要了解一下它是怎樣產(chǎn)生的纸厉。

下面展示了一個(gè)不經(jīng)意產(chǎn)生循環(huán)強(qiáng)引用的例子。例子定義了兩個(gè)類:Person和Apartment五嫂,用來(lái)建模公寓和它其中的居民:”

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") }
}

每一個(gè)Person實(shí)例有一個(gè)類型為String颗品,名字為name的屬性,并有一個(gè)可選的初始化為nil的apartment屬性沃缘。apartment屬性是可選的躯枢,因?yàn)橐粋€(gè)人并不總是擁有公寓。

類似的槐臀,每個(gè)Apartment實(shí)例有一個(gè)叫unit锄蹂,類型為String的屬性,并有一個(gè)可選的初始化為nil的tenant屬性峰档。tenant屬性是可選的败匹,因?yàn)橐粭澒⒉⒉豢偸怯芯用瘛?/p>

這兩個(gè)類都定義了析構(gòu)函數(shù),用以在類實(shí)例被析構(gòu)的時(shí)候輸出信息讥巡。這讓你能夠知曉Person和Apartment的實(shí)例是否像預(yù)期的那樣被銷毀掀亩。

接下來(lái)的代碼片段定義了兩個(gè)可選類型的變量john和unit4A,并分別被設(shè)定為下面的Apartment和Person的實(shí)例欢顷。這兩個(gè)變量都被初始化為nil槽棍,這正是可選類型的優(yōu)點(diǎn):

“var john: Person?
var unit4A: Apartment?

現(xiàn)在你可以創(chuàng)建特定的Person和Apartment實(shí)例并將賦值給john和unit4A變量:
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")”

“在兩個(gè)實(shí)例被創(chuàng)建和賦值后,下圖表現(xiàn)了強(qiáng)引用的關(guān)系抬驴。變量john現(xiàn)在有一個(gè)指向Person實(shí)例的強(qiáng)引用炼七,而變量unit4A有一個(gè)指向Apartment實(shí)例的強(qiáng)引用:


現(xiàn)在你能夠?qū)⑦@兩個(gè)實(shí)例關(guān)聯(lián)在一起,這樣人就能有公寓住了布持,而公寓也有了房客豌拙。注意感嘆號(hào)是用來(lái)展開和訪問(wèn)可選變量john和unit4A中的實(shí)例,這樣實(shí)例的屬性才能被賦值:”

“john!.apartment = unit4A
unit4A!.tenant = john
在將兩個(gè)實(shí)例聯(lián)系在一起之后题暖,強(qiáng)引用的關(guān)系如圖所示:”

“不幸的是按傅,這兩個(gè)實(shí)例關(guān)聯(lián)后會(huì)產(chǎn)生一個(gè)循環(huán)強(qiáng)引用。Person實(shí)例現(xiàn)在有了一個(gè)指向Apartment實(shí)例的強(qiáng)引用胧卤,而Apartment實(shí)例也有了一個(gè)指向Person實(shí)例的強(qiáng)引用唯绍。因此,當(dāng)你斷開john和unit4A變量所持有的強(qiáng)引用時(shí)枝誊,引用計(jì)數(shù)并不會(huì)降為0况芒,實(shí)例也不會(huì)被 ARC 銷毀:”


“john = nil
unit4A = nil”

“注意,當(dāng)你把這兩個(gè)變量設(shè)為nil時(shí)叶撒,沒(méi)有任何一個(gè)析構(gòu)函數(shù)被調(diào)用绝骚。循環(huán)強(qiáng)引用會(huì)一直阻止Person和Apartment類實(shí)例的銷毀耐版,這就在你的應(yīng)用程序中造成了內(nèi)存泄漏。
在你將john和unit4A賦值為nil后皮壁,強(qiáng)引用關(guān)系如下圖:”


“Person和Apartment實(shí)例之間的強(qiáng)引用關(guān)系保留了下來(lái)并且不會(huì)被斷開椭更。

3.解決實(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)引用。

當(dāng)其他的實(shí)例有更短的生命周期時(shí)湿滓,使用弱引用滴须,也就是說(shuō),當(dāng)其他實(shí)例析構(gòu)在先時(shí)叽奥。在上面公寓的例子中扔水,很顯然一個(gè)公寓在它的生命周期內(nèi)會(huì)在某個(gè)時(shí)間段沒(méi)有它的主人,所以一個(gè)弱引用就加在公寓類里面朝氓,避免循環(huán)引用魔市。相比之下,當(dāng)其他實(shí)例有相同的或者更長(zhǎng)生命周期時(shí)赵哲,請(qǐng)使用無(wú)主引用待德。”

弱引用

弱引用不會(huì)對(duì)其引用的實(shí)例保持強(qiáng)引用枫夺,因而不會(huì)阻止 ARC 銷毀被引用的實(shí)例将宪。這個(gè)特性阻止了引用變?yōu)檠h(huán)強(qiáng)引用。聲明屬性或者變量時(shí)橡庞,在前面加上weak關(guān)鍵字表明這是一個(gè)弱引用较坛。

因?yàn)槿跻貌粫?huì)保持所引用的實(shí)例,即使引用存在扒最,實(shí)例也有可能被銷毀丑勤。因此,ARC 會(huì)在引用的實(shí)例被銷毀后自動(dòng)將其賦值為nil吧趣。并且因?yàn)槿跻每梢栽试S它們的值在運(yùn)行時(shí)被賦值為nil法竞,所以它們會(huì)被定義為可選類型變量,而不是常量再菊。

你可以像其他可選值一樣爪喘,檢查弱引用的值是否存在,你將永遠(yuǎn)不會(huì)訪問(wèn)已銷毀的實(shí)例的引用纠拔。

“注意
當(dāng) ARC 設(shè)置弱引用為nil時(shí)秉剑,屬性觀察不會(huì)被觸發(fā)〕砘澹”

下面的例子跟上面Person和Apartment的例子一致侦鹏,但是有一個(gè)重要的區(qū)別诡曙。這一次,Apartment的tenant屬性被聲明為弱引用:

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") }
}

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

var john: Person?
var unit4A: Apartment?
 
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
 
john!.apartment = unit4A
unit4A!.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
// 打印 “John Appleseed is being deinitialized

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

unit4A = nil
// Prints "Apartment 4A is being deinitialized

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


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

“注意
在使用垃圾收集的系統(tǒng)里捷绒,弱指針有時(shí)用來(lái)實(shí)現(xiàn)簡(jiǎn)單的緩沖機(jī)制,因?yàn)闆](méi)有強(qiáng)引用的對(duì)象只會(huì)在內(nèi)存壓力觸發(fā)垃圾收集時(shí)才被銷毀贯要。但是在 ARC 中暖侨,一旦值的最后一個(gè)強(qiáng)引用被移除,就會(huì)被立即銷毀郭毕,這導(dǎo)致弱引用并不適合上面的用途它碎。”

無(wú)主引用

“和弱引用類似显押,無(wú)主引用不會(huì)牢牢保持住引用的實(shí)例扳肛。和弱引用不同的是,無(wú)主引用在其他實(shí)例有相同或者更長(zhǎng)的生命周期時(shí)使用乘碑。你可以在聲明屬性或者變量時(shí)挖息,在前面加上關(guān)鍵字unowned表示這是一個(gè)無(wú)主引用。

無(wú)主引用通常都被期望擁有值兽肤。不過(guò) ARC 無(wú)法在實(shí)例被銷毀后將無(wú)主引用設(shè)為nil套腹,因?yàn)榉强蛇x類型的變量不允許被賦值為nil∽收。”

“重要
使用無(wú)主引用电禀,你必須確保引用始終指向一個(gè)未銷毀的實(shí)例。
如果你試圖在實(shí)例被銷毀后笤休,訪問(wèn)該實(shí)例的無(wú)主引用尖飞,會(huì)觸發(fā)運(yùn)行時(shí)錯(cuò)誤。”

下面的例子定義了兩個(gè)類政基,CustomerCreditCard贞铣,模擬了銀行客戶和客戶的信用卡。這兩個(gè)類中沮明,每一個(gè)都將另外一個(gè)類的實(shí)例作為自身的屬性辕坝。這種關(guān)系可能會(huì)造成循環(huán)強(qiáng)引用。

CustomerCreditCard之間的關(guān)系與前面弱引用例子中ApartmentPerson的關(guān)系略微不同荐健。在這個(gè)數(shù)據(jù)模型中酱畅,一個(gè)客戶可能有或者沒(méi)有信用卡,但是一張信用卡總是關(guān)聯(lián)著一個(gè)客戶摧扇。為了表示這種關(guān)系圣贸,Customer類有一個(gè)可選類型的card屬性挚歧,但是CreditCard類有一個(gè)非可選類型的customer屬性扛稽。

此外,只能通過(guò)將一個(gè)number值和customer實(shí)例傳遞給CreditCard構(gòu)造函數(shù)的方式來(lái)創(chuàng)建CreditCard實(shí)例滑负。這樣可以確保當(dāng)創(chuàng)建CreditCard實(shí)例時(shí)總是有一個(gè)customer實(shí)例與之關(guān)聯(lián)在张。

由于信用卡總是關(guān)聯(lián)著一個(gè)客戶,因此將customer屬性定義為無(wú)主引用矮慕,用以避免循環(huán)強(qiáng)引用:

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屬性的存儲(chǔ)量在 32 位和 64 位系統(tǒng)上都能足夠容納 16 位的卡號(hào)。

下面的代碼片段定義了一個(gè)叫john的可選類型Customer變量痴鳄,用來(lái)保存某個(gè)特定客戶的引用瘟斜。由于是可選類型,所以變量被初始化為nil:

var john: Customer?

現(xiàn)在你可以創(chuàng)建Customer類的實(shí)例痪寻,用它初始化CreditCard實(shí)例螺句,并將新創(chuàng)建的CreditCard實(shí)例賦值為客戶的card屬性:

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

在你關(guān)聯(lián)兩個(gè)實(shí)例后,它們的引用關(guān)系如下圖所示:


Customer實(shí)例持有對(duì)CreditCard實(shí)例的強(qiáng)引用橡类,而CreditCard實(shí)例持有對(duì)Customer實(shí)例的無(wú)主引用蛇尚。

由于customer的無(wú)主引用,當(dāng)你斷開john變量持有的強(qiáng)引用時(shí)顾画,再也沒(méi)有指向Customer實(shí)例的強(qiáng)引用了:


由于再也沒(méi)有指向Customer實(shí)例的強(qiáng)引用取劫,該實(shí)例被銷毀了。其后研侣,再也沒(méi)有指向CreditCard實(shí)例的強(qiáng)引用谱邪,該實(shí)例也隨之被銷毀了:

john = nil
// 打印 “John Appleseed is being deinitialized”
// 打印 ”Card #1234567890123456 is being deinitialized”

最后的代碼展示了在john變量被設(shè)為nil后Customer實(shí)例和CreditCard實(shí)例的構(gòu)造函數(shù)都打印出了“銷毀”的信息。

注意
上面的例子展示了如何使用安全的無(wú)主引用庶诡。對(duì)于需要禁用運(yùn)行時(shí)的安全檢查的情況(例如惦银,出于性能方面的原因),Swift還提供了不安全的無(wú)主引用。與所有不安全的操作一樣璧函,你需要負(fù)責(zé)檢查代碼以確保其安全性傀蚌。 你可以通過(guò)unowned(unsafe)來(lái)聲明不安全無(wú)主引用。如果你試圖在實(shí)例被銷毀后蘸吓,訪問(wèn)該實(shí)例的不安全無(wú)主引用善炫,你的程序會(huì)嘗試訪問(wèn)該實(shí)例之前所在的內(nèi)存地址,這是一個(gè)不安全的操作库继。

無(wú)主引用以及隱式解析可選屬性

上面弱引用和無(wú)主引用的例子涵蓋了兩種常用的需要打破循環(huán)強(qiáng)引用的場(chǎng)景箩艺。
Person和Apartment的例子展示了兩個(gè)屬性的值都允許為nil,并會(huì)潛在的產(chǎn)生循環(huán)強(qiáng)引用宪萄。這種場(chǎng)景最適合用弱引用來(lái)解決艺谆。

Customer和CreditCard的例子展示了一個(gè)屬性的值允許為nil,而另一個(gè)屬性的值不允許為nil拜英,這也可能會(huì)產(chǎn)生循環(huán)強(qiáng)引用静汤。這種場(chǎng)景最適合通過(guò)無(wú)主引用來(lái)解決【有祝”
“然而虫给,存在著第三種場(chǎng)景,在這種場(chǎng)景中侠碧,兩個(gè)屬性都必須有值抹估,并且初始化完成后永遠(yuǎn)不會(huì)為nil。在這種場(chǎng)景中弄兜,需要一個(gè)類使用無(wú)主屬性药蜻,而另外一個(gè)類使用隱式解析可選屬性。

這使兩個(gè)屬性在初始化完成后能被直接訪問(wèn)(不需要可選展開)替饿,同時(shí)避免了循環(huán)引用语泽。這一節(jié)將為你展示如何建立這種關(guān)系。

下面的例子定義了兩個(gè)類盛垦,Country和City湿弦,每個(gè)類將另外一個(gè)類的實(shí)例保存為屬性。在這個(gè)模型中腾夯,每個(gè)國(guó)家必須有首都颊埃,每個(gè)城市必須屬于一個(gè)國(guó)家。為了實(shí)現(xiàn)這種關(guān)系蝶俱,Country類擁有一個(gè)capitalCity屬性班利,而City類有一個(gè)country屬性:

class Country: NSObject {
    
    var name: String?
    var capitalCity: City!
    
    override init() {
        super.init()
    }
  
    init(name: String, capitalName: String) {
        super.init()
        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
    }
}

為了建立兩個(gè)類的依賴關(guān)系榨呆,City的構(gòu)造函數(shù)接受一個(gè)Country實(shí)例作為參數(shù)罗标,并且將實(shí)例保存到country屬性。
Country的構(gòu)造函數(shù)調(diào)用了City的構(gòu)造函數(shù)。然而闯割,只有Country的實(shí)例完全初始化后彻消,Country的構(gòu)造函數(shù)才能把self傳給City的構(gòu)造函數(shù)。在兩段式構(gòu)造過(guò)程中有具體描述宙拉。

為了滿足這種需求宾尚,通過(guò)在類型結(jié)尾處加上感嘆號(hào)(City!)的方式,將Country的capitalCity屬性聲明為隱式解析可選類型的屬性谢澈。這意味著像其他可選類型一樣煌贴,capitalCity屬性的默認(rèn)值為nil吩案,但是不需要展開它的值就能訪問(wèn)它影钉。在隱式解析可選類型中有描述。

由于capitalCity默認(rèn)值為nil签财,一旦Country的實(shí)例在構(gòu)造函數(shù)中給name屬性賦值后敬鬓,整個(gè)初始化過(guò)程就完成了淹朋。這意味著一旦name屬性被賦值后,Country的構(gòu)造函數(shù)就能引用并傳遞隱式的self列林。Country的構(gòu)造函數(shù)在賦值capitalCity時(shí)瑞你,就能將self作為參數(shù)傳遞給City的構(gòu)造函數(shù)。

以上的意義在于你可以通過(guò)一條語(yǔ)句同時(shí)創(chuàng)建Country和City的實(shí)例希痴,而不產(chǎn)生循環(huán)強(qiáng)引用,并且capitalCity的屬性能被直接訪問(wèn)春感,而不需要通過(guò)感嘆號(hào)來(lái)展開它的可選值:

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

在上面的例子中砌创,使用隱式解析可選值意味著滿足了類的構(gòu)造函數(shù)的兩個(gè)構(gòu)造階段的要求。capitalCity屬性在初始化完成后鲫懒,能像非可選值一樣使用和存取嫩实,同時(shí)還避免了循環(huán)強(qiáng)引用】遥”

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末甲献,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子颂翼,更是在濱河造成了極大的恐慌晃洒,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,104評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件朦乏,死亡現(xiàn)場(chǎng)離奇詭異球及,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)呻疹,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門吃引,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事镊尺‰澹” “怎么了?”我有些...
    開封第一講書人閱讀 168,697評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵庐氮,是天一觀的道長(zhǎng)吕粗。 經(jīng)常有香客問(wèn)我,道長(zhǎng)旭愧,這世上最難降的妖魔是什么颅筋? 我笑而不...
    開封第一講書人閱讀 59,836評(píng)論 1 298
  • 正文 為了忘掉前任,我火速辦了婚禮输枯,結(jié)果婚禮上议泵,老公的妹妹穿的比我還像新娘。我一直安慰自己桃熄,他們只是感情好先口,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瞳收,像睡著了一般碉京。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上螟深,一...
    開封第一講書人閱讀 52,441評(píng)論 1 310
  • 那天谐宙,我揣著相機(jī)與錄音,去河邊找鬼界弧。 笑死凡蜻,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的垢箕。 我是一名探鬼主播划栓,決...
    沈念sama閱讀 40,992評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼条获!你這毒婦竟也來(lái)了忠荞?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,899評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤帅掘,失蹤者是張志新(化名)和其女友劉穎委煤,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體锄开,經(jīng)...
    沈念sama閱讀 46,457評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡素标,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評(píng)論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了萍悴。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片头遭。...
    茶點(diǎn)故事閱讀 40,664評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡寓免,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出计维,到底是詐尸還是另有隱情袜香,我是刑警寧澤,帶...
    沈念sama閱讀 36,346評(píng)論 5 350
  • 正文 年R本政府宣布鲫惶,位于F島的核電站蜈首,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏欠母。R本人自食惡果不足惜欢策,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望赏淌。 院中可真熱鬧踩寇,春花似錦、人聲如沸六水。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)掷贾。三九已至睛榄,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間想帅,已是汗流浹背场靴。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留博脑,地道東北人憎乙。 一個(gè)月前我還...
    沈念sama閱讀 49,081評(píng)論 3 377
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像叉趣,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子该押,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容