Swift使用自動引用計數(shù)(ARC)
機制來跟蹤和管理你的應(yīng)用程序的內(nèi)存。通常情況下褐缠,Swift 內(nèi)存管理機制會一直起作用政鼠,你無須自己來考慮內(nèi)存的管理。ARC 會在類的實例不再被使用時队魏,自動釋放其占用的內(nèi)存公般。
然而在少數(shù)情況下,為了能幫助你管理內(nèi)存胡桨,ARC 需要更多的官帘,代碼之間關(guān)系的信息。本章描述了這些情況昧谊,并 且為你示范怎樣才能使 ARC 來管理你的應(yīng)用程序的所有內(nèi)存刽虹。在 Swift 使用 ARC 與在 Obejctive-C 中使用 ARC 非常類似。
自動引用計數(shù)的工作機制
當你每次創(chuàng)建一個類的新的實例的時候呢诬,ARC 會分配一塊內(nèi)存來儲存該實例信息涌哲。內(nèi)存中會包含實例的類型信 息,以及這個實例所有相關(guān)的存儲型屬性的值尚镰。
此外膛虫,當實例不再被使用時,ARC 釋放實例所占用的內(nèi)存钓猬,并讓釋放的內(nèi)存能挪作他用。這確保了不再被使用的 實例撩独,不會一直占用內(nèi)存空間敞曹。
然而账月,當 ARC 收回和釋放了正在被使用中的實例,該實例的屬性和方法將不能再被訪問和調(diào)用澳迫。實際上局齿,如果你 試圖訪問這個實例,你的應(yīng)用程序很可能會崩潰橄登。
為了確保使用中的實例不會被銷毀抓歼,ARC 會跟蹤和計算每一個實例正在被多少屬性,常量和變量所引用拢锹。哪怕實 例的引用數(shù)為1谣妻,ARC都不會銷毀這個實例。
為了使上述成為可能卒稳,無論你將實例賦值給屬性蹋半、常量或變量,它們都會創(chuàng)建此實例的強引用充坑。之所以稱之
為“強”引用减江,是因為它會將實例牢牢地保持住,只要強引用還在捻爷,實例是不允許被銷毀的辈灼。
類實例之間的循環(huán)強引用
class Person{
let name : String;
init(name : String) {
self.name = name;
}
var apertment : Apartment?;
}
class Apartment {
let unit : String;
init(unit : String) {
self.unit = unit;
}
var tenant : Person?;
}
var john : Person?;
var unit4A : Apartment?;
john = Person(name: "John");
unit4A = Apartment(unit: "4a");
//強引用
john!.apertment = unit4A;
unit4A!.tenant = john;
變量john
現(xiàn)在有一個指向 Person
實例的強引用
,而變 量unit4A
有一個指向Apartment
實例的強引用
也榄。
這兩個實例關(guān)聯(lián)后會產(chǎn)生一個循環(huán)強引用巡莹。 Person 實例現(xiàn)在有了一個指向 Apartment 實例的強引 用,而 Apartment 實例也有了一個指向 Person 實例的強引用手蝎。因此榕莺,當你斷開 john 和 unit4A 變量所持有的強 引用時,引用計數(shù)并不會降為 0 棵介,實例也不會被 ARC 銷毀钉鸯。
john = nil
unit4A = nil
當你把這兩個變量設(shè)為nil
時,沒有任何一個析構(gòu)函數(shù)
被調(diào)用邮辽。循環(huán)強引用會一直阻止 Person 和 Apartme nt 類實例的銷毀唠雕,這就在你的應(yīng)用程序中造成了內(nèi)存泄漏
。
解決實例之間的循環(huán)強引用
Swift提供了兩種辦法來解決你在使用類的屬性時所遇到的循環(huán)強引用問題:弱引用
和無主引用
吨述。
弱引用
和無主引用
允許循環(huán)引用中的一個實例引用而另外一個實例不保持強引用岩睁。這樣實例能夠互相引用而不產(chǎn)生循環(huán)強引用。
- 弱引用
弱引用不會對其引用的實例保持強引用揣云,因為不會阻止ARC銷毀被引用的實例捕儒。這個特性阻止了引用變?yōu)檠h(huán)強引用。申明屬性或者變量時,在前面加上weak
關(guān)鍵字表明這是一個弱引用刘莹。
因為弱引用不會保持引用的實例阎毅,即使引用存在,實例也有可能被銷毀点弯,因此扇调,ARC會在引用的實例被銷毀后自動將其賦值為nil
,并且因為弱引用可以允許他們的值在運行時被賦值為nil
抢肛,所以他們會被定義為可選類型變量狼钮,而不是常量。
//解決強引用問題
class Person{
let name : String;
init(name : String) {
self.name = name;
}
var apertment : Apartment?;
}
class Apartment {
let unit : String;
init(unit : String) {
self.unit = unit;
}
weak var tenant : Person?;//設(shè)置為weak類型
}
var john : Person?;
var unit4A : Apartment?;
john = Person(name: "John");
unit4A = Apartment(unit: "4a");
john!.apertment = unit4A;
unit4A!.tenant = john;
- 無主引用
和弱引用類似捡絮,無主引用不會牢牢保持住引用的實例熬芜。和弱引用不同的是,無主引用在其他實例有相同或者更長的生命周期時使用锦援,你可以在聲明屬性或者變量時猛蔽,在前面加上關(guān)鍵字unowned
表示這是一個無主引用。
無主引用通常都被期望擁有值灵寺,不過ARC無法再實例被銷毀后將無主引用設(shè)置為nil
曼库,因為非可選類型的變量不允許被賦值為nil
。(使用無主引用略板,你必須確保引用始終指向一個未銷毀的實例毁枯。)。