題目中的代碼存在可能循環(huán)引用的問題蜕着,對(duì)象的循環(huán)引用會(huì)造成ARC引用計(jì)數(shù)無(wú)法釋放被引用的任何一個(gè)對(duì)象痘拆,從而造成內(nèi)存泄露染簇。
上述代碼中Customer類包含一個(gè)CreditCard類的實(shí)例屬性蜒简,而CreditCard類也包含了一個(gè)Customer類的實(shí)例屬性休溶。一旦在堆空間中這兩個(gè)類的實(shí)例相互引用對(duì)方就會(huì)造成循環(huán)引用蚤氏。
實(shí)例代碼:生成Customer類和CreditCard類的實(shí)例屬性甘耿,并讓這兩個(gè)實(shí)例屬性互相引用。這樣會(huì)造成循環(huán)引用竿滨,導(dǎo)致內(nèi)存泄露佳恬。
var person:Customer
var bankCard:CreditCard
person = Customer(name:"Peter")
bankCard = CreditCard(number:111000222333, customer:person)
person.card = bankCard //這樣便形成了循環(huán)引用
//由于形成了循環(huán)引用,下面兩句代碼執(zhí)行之后依然不會(huì)釋放堆里面的內(nèi)存空間
person = nil
bankCard = nil
解決循環(huán)引用造成內(nèi)存泄露有三種方式:
1于游、在合適的地方毁葱,手動(dòng)將循環(huán)引用解除
var person:Customer
var bankCard:CreditCard
person = Customer(name:"Peter")
bankCard = CreditCard(number:111000222333, customer:person)
person.card = bankCard //這樣便形成了循環(huán)引用
person.card = nil //在釋放兩個(gè)實(shí)例對(duì)象前可以手動(dòng)的釋放person對(duì)CreditCard類實(shí)例的引用
//由于上面一句代碼斷開了循環(huán)引用,下面兩句代碼執(zhí)行之后就會(huì)釋放堆里面的內(nèi)存空間
person = nil
bankCard = nil
2贰剥、可將引用聲明為弱引用
class Customer {
let name:String
weak var card:CreditCard? //設(shè)置成弱引用倾剿,弱引用使用時(shí)將不再讓ARC計(jì)數(shù)加一,如果CreditCard的實(shí)例提前釋放的話蚌成,屬性將變成nil
init(name:String) {
self.name = name
}
deinit {(print("\(name) is being deinitialized")}
}
3前痘、如果不允許對(duì)象引用為nil時(shí),可以將對(duì)象聲明為無(wú)主引用
class Customer {
let name:String
unowned var card:CreditCard //設(shè)置無(wú)主引用担忧,無(wú)主引用使用時(shí)將不再讓ARC計(jì)數(shù)加一际度,如果CreditCard的實(shí)例提前釋放的話,屬性將依然指向那個(gè)地址涵妥。需要確保不要非法訪問,不然會(huì)拋出異常
init(name:String) {
self.name = name
}
deinit {(print("\(name) is being deinitialized")}
}
上述三種解決方案我覺得方案二坡锡,即使用弱引用最適合題目情況蓬网,因?yàn)楸旧鞢ustomer類中的屬性card可以允許為空,如果使用手動(dòng)循環(huán)的話有可能遺忘鹉勒。第二種方法保證了不會(huì)出現(xiàn)循環(huán)引用而導(dǎo)致內(nèi)存泄露帆锋。第三種方法使用無(wú)主引用,第一是改變了Customer類中的屬性card的可為空屬性禽额,第二當(dāng)card屬性指向的堆空間提前釋放的話锯厢,會(huì)有非法訪問的風(fēng)險(xiǎn)。所以使用弱引用決絕方案最好脯倒。