swift 中如何解決循環(huán)引用
-
解決類(lèi)之間的循環(huán)引用
weak:弱引用, 必須為變量(var)蜂莉,可選類(lèi)型(?)蜡娶,指針指向的對(duì)象在銷(xiāo)毀后,指針指向nil映穗,再次向該對(duì)象發(fā)送消息不會(huì)引起崩潰(錯(cuò)誤)
unovned:無(wú)主引用窖张,非可選類(lèi)型(不可為?)男公,指針指向的對(duì)象在銷(xiāo)毀后荤堪,指針并不會(huì)指向nil,相當(dāng)于一個(gè)野指針枢赔,再次向該對(duì)象發(fā)送消息不會(huì)引起崩潰(錯(cuò)誤)
-
weak和unowned 使用規(guī)則:
1.兩個(gè)實(shí)例的屬性引用都可為nil(澄阳?可選的),其中一個(gè)用(weak)弱引用。
2.其中一個(gè)屬性的引用可以為nil(踏拜?可選的),另一個(gè)屬性引用不允許為nil碎赢,不允許為空的使用(unowned)無(wú)主引用
3.兩個(gè)實(shí)例的屬性都必須有值,并且初始化后永遠(yuǎn)不會(huì)為nil.這種場(chǎng)景中速梗,需要一個(gè)類(lèi)使用(unowned)無(wú)主屬性肮塞,另一個(gè)使用隱式解析可選(!)屬性
-
使用場(chǎng)景
設(shè)置delegate時(shí)襟齿,我們常使用 weak
weak var delegate? : Protocol
-
解決閉包引起的循環(huán)引用
-
產(chǎn)生的原因
循環(huán)引用的產(chǎn)生,是因?yàn)殚]包和類(lèi)相似枕赵,都是引用類(lèi)型猜欺。
-
解決辦法
解決閉包引起的循環(huán)引用的方法是定義捕獲列表(數(shù)組),捕獲列表中的每一項(xiàng)都由一對(duì)元素組成拷窜,一個(gè)元素是weak或unowned關(guān)鍵字开皿,另一個(gè)元素是類(lèi)實(shí)例的引用。
1.在閉包和捕獲的實(shí)例總是相互引用并且總是同時(shí)銷(xiāo)毀時(shí)篮昧,將閉包內(nèi)的捕獲定義為無(wú)主引用
2.在捕獲的引用可能變?yōu)閚il時(shí)赋荆,將閉包內(nèi)的捕獲定義為弱引用。弱引用總是可選類(lèi)型懊昨,并且當(dāng)引用的實(shí)例被銷(xiāo)毀后窄潭,弱引用的值會(huì)自動(dòng)置為nil。(至于到底該使用weak還是unowned酵颁,我們要根據(jù)實(shí)際使用情況來(lái)選擇)
-
使用場(chǎng)景
在self屬性存儲(chǔ)為閉包時(shí)嫉你,閉包中擁有對(duì)self的引用時(shí),捕獲列表定義為無(wú)主引用(unowned)
class People: NSObject {
let lastName:String
let firstName:String
//這里使用無(wú)主引用
lazy var name:String = {
[unowned self] in
return self.firstName + self.lastName
}()
init(firstName:String,lastName:String) {
self.firstName = firstName
self.lastName = lastName
}
deinit {
print("被銷(xiāo)毀")
}
}
-
疑問(wèn)
在上例的閉包中材义,使有weak引用也是沒(méi)問(wèn)題的均抽。那我是不是可以在所有的捕獲列表中都用weak引用,來(lái)避免由于使用無(wú)主引用而導(dǎo)致的坑能崩潰的問(wèn)題其掂?希望明白的人給予指導(dǎo)油挥。謝謝
下面的內(nèi)容由OC內(nèi)存管理總結(jié)而來(lái),點(diǎn)擊查看原文款熬,感謝博主的總結(jié)深寥。
ARC內(nèi)存管理機(jī)制
-
ARC的判斷準(zhǔn)則:
只要沒(méi)有強(qiáng)指針指(strong)向?qū)ο螅瑢?duì)象就會(huì)被釋放贤牛。
-
指針?lè)诸?lèi)
(1)強(qiáng)指針:默認(rèn)的情況下惋鹅,所有的指針都是強(qiáng)指針,關(guān)鍵字strong
(2)弱指針:_ _weak關(guān)鍵字修飾的指針
聲明一個(gè)弱指針如下:
_ _weak Person *p;
ARC中殉簸,只要弱指針指向的對(duì)象不在了闰集,就直接把弱指針做清空(置nil)操作。(向空指針發(fā)送消息不回崩潰(報(bào)錯(cuò)))般卑。
-
@property說(shuō)明
ARC中在property處不再使用retain,而是使用strong武鲁,在dealloc方法中不需要再[super dealloc]。
@property(nonatomic,strong)Dog *dog;// 意味著生成的成員變量dog是一個(gè)強(qiáng)指針蝠检,相當(dāng)于以前的retain沐鼠。
如果換成是弱指針,則換成weak,不需要加 _饲梭。
-
ARC的特點(diǎn)總結(jié):
(1)不允許調(diào)用release乘盖,retain,retainCount
(2)允許重寫(xiě)dealloc,但是不允許調(diào)用[super dealloc]
(3)@property的參數(shù):
Strong:相當(dāng)于原來(lái)的retain(適用于OC對(duì)象類(lèi)型)憔涉,成員變量是強(qiáng)指針
Weak:相當(dāng)于原來(lái)的assign,(適用于oc對(duì)象類(lèi)型)订框,成員變量是弱指針
Assign:適用于非OC對(duì)象類(lèi)型(基礎(chǔ)類(lèi)型)
-
補(bǔ)充
讓程序兼容ARC和非ARC部分。轉(zhuǎn)變?yōu)榉茿RC -fno-objc-arc 轉(zhuǎn)變?yōu)锳RC的兜叨, -f-objc-arc 布蔗。
ARC也需要考慮循環(huán)引用問(wèn)題:一端使用retain,另一端使用assign浪腐。
OC內(nèi)存管理(No-ARC)
-
為什么需要內(nèi)存管理?
本質(zhì)原因是因?yàn)閷?duì)象和其他數(shù)據(jù)類(lèi)型在系統(tǒng)中的存儲(chǔ)空間不一樣顿乒,其它局部變量主要存放于棧中议街,而對(duì)象存儲(chǔ)于堆中,當(dāng)代碼塊結(jié)束時(shí)這個(gè)代碼塊中涉及的所有局部變量會(huì)被回收璧榄,指向?qū)ο蟮闹羔樢脖换厥仗劁觯藭r(shí)對(duì)象已經(jīng)沒(méi)有指針指向,但依然存在于內(nèi)存中骨杂,造成內(nèi)存泄露涂身。
理解為指針已經(jīng)收回,但是引用計(jì)數(shù)并沒(méi)有-1搓蚪,造成內(nèi)存泄漏蛤售。
-
對(duì)象的基本結(jié)構(gòu)
每個(gè)OC對(duì)象都有自己的引用計(jì)數(shù)器,是一個(gè)整數(shù)表示對(duì)象被引用的次數(shù)妒潭,即現(xiàn)在有多少東西在使用這個(gè)對(duì)象悴能。對(duì)象剛被創(chuàng)建時(shí),默認(rèn)計(jì)數(shù)器值為1雳灾,當(dāng)計(jì)數(shù)器的值變?yōu)?時(shí)漠酿,則對(duì)象銷(xiāo)毀。
在每個(gè)OC對(duì)象內(nèi)部谎亩,都專(zhuān)門(mén)有4個(gè)字節(jié)的存儲(chǔ)空間來(lái)存儲(chǔ)引用計(jì)數(shù)器炒嘲。
-
操作
給對(duì)象發(fā)送消息,進(jìn)行相應(yīng)的計(jì)數(shù)器操作匈庭。
Retain消息:使計(jì)數(shù)器+1夫凸,改方法返回對(duì)象本身
Release消息:使計(jì)數(shù)器-1(并不代表釋放對(duì)象)
retainCount消息:獲得對(duì)象當(dāng)前的引用計(jì)數(shù)器值
-
對(duì)象的銷(xiāo)毀
當(dāng)對(duì)象被銷(xiāo)毀時(shí),系統(tǒng)會(huì)自動(dòng)向?qū)ο蟀l(fā)送一條dealloc消息嚎花,一般會(huì)重寫(xiě)dealloc方法寸痢,在這里釋放相關(guān)的資源,一旦重寫(xiě)了dealloc方法就必須調(diào)用[super dealloc]紊选,并且放在代碼塊的最后調(diào)用(不能直接調(diào)用dealloc方法)啼止。一旦對(duì)象被回收了道逗,那么他所占據(jù)的存儲(chǔ)空間就不再可用,堅(jiān)持使用會(huì)導(dǎo)致程序崩潰(野指針錯(cuò)誤)献烦。
-
相關(guān)概念
野指針:所指對(duì)象已經(jīng)不存在的指針
空指針:沒(méi)有指向任何東西的指針
訪(fǎng)問(wèn)野指針會(huì)崩潰滓窍,訪(fǎng)問(wèn)空指針不會(huì)崩潰(報(bào)錯(cuò))
-
內(nèi)存管理原則
-
原則
-
只要還有人在使用某個(gè)對(duì)象,那么這個(gè)對(duì)象就不會(huì)被回收巩那;
只要你想使用這個(gè)對(duì)象吏夯,那么就應(yīng)該讓這個(gè)對(duì)象的引用計(jì)數(shù)器+1;
當(dāng)你不想使用這個(gè)對(duì)象時(shí)即横,應(yīng)該讓對(duì)象的引用計(jì)數(shù)器-1噪生;
-
誰(shuí)創(chuàng)建,誰(shuí)release
(1)如果你通過(guò)alloc,new,copy來(lái)創(chuàng)建了一個(gè)對(duì)象东囚,引用計(jì)數(shù)都會(huì)+1跺嗽,那么你就必須調(diào)用release或者autorelease方法
(2)不是你創(chuàng)建的就不用你去負(fù)責(zé)
-
誰(shuí)retain,誰(shuí)release
-
總結(jié)
誰(shuí)創(chuàng)建(alloc,new,copy)页藻,誰(shuí)release,誰(shuí)retain桨嫁,誰(shuí)release
-
@property的參數(shù)
內(nèi)存管理相關(guān)參數(shù)
Retain:對(duì)對(duì)象release舊值,retain新值(適用于OC對(duì)象類(lèi)型)
Assign:直接賦值(默認(rèn)份帐,適用于非oc對(duì)象類(lèi)型)
Copy:release舊值璃吧,copy新值
-
內(nèi)存管理中的循環(huán)引用問(wèn)題以及解決
案例:每個(gè)人有一張身份證,每張身份證對(duì)應(yīng)一個(gè)人废境,不能使用#import的方式相互包含畜挨,這就形成了循環(huán)引用。
新的關(guān)鍵字:@class 類(lèi)名彬坏;——解決循環(huán)引用問(wèn)題朦促,提高性能
@class僅僅告訴編譯器,在進(jìn)行編譯的時(shí)候把后面的名字作為一個(gè)類(lèi)來(lái)處理栓始。
(1)@class的作用:聲明一個(gè)類(lèi)务冕,告訴編譯器某個(gè)名稱(chēng)是一個(gè)類(lèi)
(2)開(kāi)發(fā)中引用一個(gè)類(lèi)的規(guī)范
1)在.h文件中使用@class來(lái)聲明類(lèi)
2)在.m文件中真正要使用到的時(shí)候,使用#import來(lái)包含類(lèi)中的所有東西
(3)兩端循環(huán)引用的解決方法
一端使用retain,一端使用assign(使用assign的在dealloc中也不用再release)