為什么block使用copy(ARC下也可以使用strong)
為的是將block拷貝到堆上來
// MRC
1汁咏、block類似于函數(shù)指針作媚,但是有時內(nèi)聯(lián)的(代碼直接插入到調(diào)用者處,免去了普通函數(shù)調(diào)用的過程漂问,效率更高)
2、block是一個代碼片段蚤假,在OC中被看作是OC對象
3吧兔、block建議使用copy或者strong策略
4、block分為三種__NSGlobalBlock__掩驱,__NSMallocBlock__冬竟,__NSStackBlock__
MRC默認(rèn)情況下是__NSGlobalBlock__民逼,使用(捕獲)了外部非static和全局的變量會變成__NSStackBlock__,__NSStackBlock__的block使用了copy策略拼苍,就變成__NSMallocBlock__的了
// ARC下
1、如果使用
ARC默認(rèn)情況下是__NSGlobalBlock__疮鲫,使用(捕獲)了外部非static和全局的變量會變成__NSStackBlock__,將__NSStackBlock__賦值給變量或者屬性(非assign妇多,使用assign還是__NSStackBlock__燕侠,使用了copy和strong都一樣)系統(tǒng)默認(rèn)將會發(fā)生copy操作到棧上,變成__NSMallocBlock__
5绢彤、在MRC和ARC下都不要使用assign策略
為什么將block拷貝到堆上來
1、block拷貝到堆上來基本是為了處理__NSStackBlock__茫舶,捕獲外部變量(因為不捕獲外部變量的時候,__NSGlobalBlock__是靜態(tài)區(qū)的讥耗,這個內(nèi)存區(qū)域是不會這幾銷毀嚷往,不存在內(nèi)存問題)
2葛账、__NSStackBlock__之所以會出現(xiàn)內(nèi)存問題是因為皮仁,在棧區(qū)內(nèi)存是系統(tǒng)自動管理的,會被自動釋放掉(計數(shù)器為0的時候趋急,以保證內(nèi)存不至于太過消耗势誊,甚至爆棧)呜达;自動釋放后block的本質(zhì)是一個結(jié)構(gòu)體指針粟耻,原本指向它的指針地址還是在的這就是空指針眉踱,即便我們把block賦值為nil霜威,地址還是0x0(空指針);一旦我們再區(qū)調(diào)用block無論是野指針還是指針都會crash戈泼。只有將block搞到堆上來,我們能購保證block的內(nèi)存不會被自動釋放扭倾,在我們不需要的情況下挽绩,block置為nil讓ARC自動管理內(nèi)存即可膛壹。
3琼牧、區(qū)別于OC可以給nil發(fā)送消息哀卫,OC會自動在發(fā)送消息的時候去判斷接受消息的對象是不是nil,是nil發(fā)送函數(shù)objc_msgSend會直接return的此改,block是直接去尋址一個內(nèi)存地址(void *FuncPtr)去執(zhí)行的。
block為什么需要判空
// oc
int main() {
void(^testBlock)();
testBlock();
}
// clang -rewrite-objc
int main() {
void(*testBlock)();
((void (*)(__block_impl *))((__block_impl *)testBlock)->FuncPtr)((__block_impl *)testBlock);
}
// block結(jié)構(gòu)體
struct __block_impl {
void *isa;
int Flags;
int Reserved;
void *FuncPtr;
};
上面OC
代碼直接運行會直接crash
占调。用clang
把OC
代碼翻譯成C++
代碼移剪,我們就會發(fā)現(xiàn),testBlock();
被翻譯為一個執(zhí)行返回(void (*)(__block_impl *))
類型且有一個參數(shù)(__block_impl *)testBlock
的函數(shù)((__block_impl *)testBlock)->FuncPtr
;我們看到testBlock
強制轉(zhuǎn)換成了結(jié)構(gòu)體指針__block_impl *
纵苛,block最終調(diào)用的是__block_impl
結(jié)構(gòu)體中的FuncPtr
;而此時的testBlock
并沒有賦值只是一個nil
,地址是0x0
,加上FuncPtr
是結(jié)構(gòu)體中的第四個成員變量取试,void *
占8個字節(jié),int
類型占4個字節(jié)怀吻,那么FuncPtr
的執(zhí)行地址是在0x0
的基礎(chǔ)上向后移動8 + 4 + 4
,一共16個字節(jié),就是0x10
;(在32位系統(tǒng)下void *
占4個字節(jié)蓬坡,最終尋址地址應(yīng)該是0x0c
)