- weak 修飾的指針變量,在指向的內(nèi)存地址銷毀后,會(huì)在 Runtime 的機(jī)制下,自動(dòng)置為 nil。
- _Unsafe_Unretain不會(huì)置為 nil,容易出現(xiàn) 懸垂指針,發(fā)生崩潰。但是 _Unsafe_Unretain 比 __weak 效率高。
1.assign 與weak區(qū)別
assign適用于基本數(shù)據(jù)類型,weak是適用于NSObject對(duì)象,并且是一個(gè)弱引用。
assign其實(shí)也可以用來修飾對(duì)象。那么我們?yōu)槭裁床挥盟揎棇?duì)象呢?因?yàn)楸籥ssign修飾的對(duì)象(一般編譯的時(shí)候會(huì)產(chǎn)生警告:Assigning retained object to unsafe property; object will be released after assignment)在釋放之后,指針的地址還是存在的,也就是說指針并沒有被置為nil,造成野指針。對(duì)象一般分配在堆上的某塊內(nèi)存,如果在后續(xù)的內(nèi)存分配中,剛好分到了這塊地址,程序就會(huì)崩潰掉。
那為什么可以用assign修飾基本數(shù)據(jù)類型?因?yàn)榛A(chǔ)數(shù)據(jù)類型一般分配在棧上,棧的內(nèi)存會(huì)由系統(tǒng)自己自動(dòng)處理,不會(huì)造成野指針。
weak修飾的對(duì)象在釋放之后,指針地址會(huì)被置為nil。所以現(xiàn)在一般弱引用就是用weak。weak使用場(chǎng)景:
在ARC下,在有可能出現(xiàn)循環(huán)引用的時(shí)候,往往要通過讓其中一端使用weak來解決,比如: delegate代理屬性,通常就會(huì)聲明為weak。
自身已經(jīng)對(duì)它進(jìn)行一次強(qiáng)引用,沒有必要再?gòu)?qiáng)引用一次時(shí)也會(huì)使用weak。比如:自定義 IBOutlet控件屬性一般也使用weak,當(dāng)然也可以使用strong。
2.strong 與copy的區(qū)別
strong 與copy都會(huì)使引用計(jì)數(shù)加1,但strong是兩個(gè)指針指向同一個(gè)內(nèi)存地址,copy會(huì)在內(nèi)存里拷貝一份對(duì)象,兩個(gè)指針指向不同的內(nèi)存地址
3.__block與__weak的區(qū)別
__block是用來修飾一個(gè)變量,這個(gè)變量就可以在block中被修改
__block:使用 __block修飾的變量在block代碼塊中會(huì)被retain(ARC下會(huì)retain,MRC下不會(huì)retain)
__weak:使用__weak修飾的變量不會(huì)在block代碼塊中被retain
同時(shí),在ARC下,要避免block出現(xiàn)循環(huán)引用 __weak typedof(self)weakSelf = self;
4.1 block變量定義時(shí)為什么用copy?block是放在哪里的?
block本身是像對(duì)象一樣可以retain,和release。但是,block在創(chuàng)建的時(shí)候,它的內(nèi)存是分配在棧(stack)上,可能被隨時(shí)回收,而不是在堆(heap)上。他本身的作于域是屬于創(chuàng)建時(shí)候的作用域,一旦在創(chuàng)建時(shí)候的作用域外面調(diào)用block將導(dǎo)致程序崩潰。通過copy可以把block拷貝(copy)到堆,保證block的聲明域外使用。
特別需要注意的地方就是在把block放到集合類當(dāng)中去的時(shí)候,如果直接把生成的block放入到集合類中,是無法在其他地方使用block,必須要對(duì)block進(jìn)行copy。
[array addObject:[[^{
NSLog(@"hello!");
} copy] autorelease]];
4.2 block 為什么不用strong?
block如果用到了self,就會(huì)retain self,如果是strong的話,就造成了循環(huán)引用
官方文檔:You should specify copy as the property attribute, because a block needs to be copied to keep track of its captured state outside of the original scope. This isn’t something you need to worry about when using Automatic Reference Counting, as it will happen automatically, but it’s best practice for the property attribute to show the resultant behavior