
第三周作業(yè)題目
題目中的代碼存在可能循環(huán)引用的問題,對象的循環(huán)引用會造成ARC引用計數(shù)無法釋放被引用的任何一個對象,從而造成內(nèi)存泄露。
上述代碼中Customer類包含一個CreditCard類的實例屬性,而CreditCard類也包含了一個Customer類的實例屬性。一旦在堆空間中這兩個類的實例相互引用對方就會造成循環(huán)引用。
實例代碼:生成Customer類和CreditCard類的實例屬性,并讓這兩個實例屬性互相引用。這樣會造成循環(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í)行之后依然不會釋放堆里面的內(nèi)存空間
person = nil
bankCard = nil
解決循環(huán)引用造成內(nèi)存泄露有三種方式:
1、在合適的地方,手動將循環(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 //在釋放兩個實例對象前可以手動的釋放person對CreditCard類實例的引用
//由于上面一句代碼斷開了循環(huán)引用,下面兩句代碼執(zhí)行之后就會釋放堆里面的內(nèi)存空間
person = nil
bankCard = nil
2、可將引用聲明為弱引用
class Customer {
let name:String
weak var card:CreditCard? //設(shè)置成弱引用,弱引用使用時將不再讓ARC計數(shù)加一,如果CreditCard的實例提前釋放的話,屬性將變成nil
init(name:String) {
self.name = name
}
deinit {(print("\(name) is being deinitialized")}
}
3、如果不允許對象引用為nil時,可以將對象聲明為無主引用
class Customer {
let name:String
unowned var card:CreditCard //設(shè)置無主引用,無主引用使用時將不再讓ARC計數(shù)加一,如果CreditCard的實例提前釋放的話,屬性將依然指向那個地址。需要確保不要非法訪問,不然會拋出異常
init(name:String) {
self.name = name
}
deinit {(print("\(name) is being deinitialized")}
}
上述三種解決方案我覺得方案二,即使用弱引用最適合題目情況,因為本身Customer類中的屬性card可以允許為空,如果使用手動循環(huán)的話有可能遺忘。第二種方法保證了不會出現(xiàn)循環(huán)引用而導(dǎo)致內(nèi)存泄露。第三種方法使用無主引用,第一是改變了Customer類中的屬性card的可為空屬性,第二當(dāng)card屬性指向的堆空間提前釋放的話,會有非法訪問的風(fēng)險。所以使用弱引用決絕方案最好。