Block介紹(二)記憶體管理與其他特性

2021-06-18 19:13:45 字數 2156 閱讀 7708

[+]

一、block放在**

我們針對不同情況來討論block的存放位置:

1.棧和堆

以下情況中的block位於堆中:12

3456

78910

1112

1314

1516

17voidfoo()

;//blk在棧裡

blkinheap = block_copy(blk);//blkinheap在堆裡

}- (void)foobar

;void(^oblkinheap)(void) = [oblkcopy];//oblkinheap在堆中

}2.全域性區

以下情況中的block位於全域性區: 1

2345

6789

10staticint(^maxintblock)(int,int) = ^(inta, intb);

- (void)foobar

voidfoo()

需要注意的是,這裡複製過後的block依舊位於全域性區,實際上,複製操作是直接返回了原block物件。

二、block引用的變數在**

1.全域性區

全域性區的變數儲存位置與block無關:12

3456

78staticintgvar =0;

//__block static int gmvar = 1;

voidfoo()

注意,static變數是不允許新增__block標記的

2.堆疊

此時,你可能會問,當函式foo返回後,棧上的j已經**,那麼blkinheap怎麼能繼續使用它?這是因為沒有__block標記的變數,會被當做實參傳入block的底層實現函式中,當block中的**被執行時,j已經不是原來的j了,所謂物是人非就是這樣吧~

另外,如果使用到變數j的所有block都沒有被複製至heap,那麼這個變數j也不會被複製至heap。

因此,即使將j++這一句放到blk()這句之前,這段**執行後,控制台列印結果也是:1024, 1。而不是1024, 2

三、其他特性

1.複製的行為

對block呼叫複製,有以下幾種情況:

1.對全域性區的block呼叫copy,會返回原指標,並且這期間不處理任何東西(至少目前的內部實現是這樣);

2.對棧上的block呼叫copy,每次會返回新複製到堆上的block的指標,同時,所有__block變數都會被複製至堆乙份(多次拷貝,只會生成乙份)。

3.對已經位於heap上的block,再次呼叫copy,只會增加block的引用計數。

為什麼我們不討論retian的行為?原因是並沒有block_retain()這樣的函式,而且objc裡面的retain訊息傳送給block物件後,其內部實現是什麼都不做。

2.objc類中的block複製

objc類例項方法中的block如果被複製至heap,那麼當前例項會被增加引用計數,當這個block被釋放時,此例項會被減少引用計數。

但如果這個block沒有使用當前例項的任何成員,那麼當前例項不會被增加引用計數。這也是很自然的道理,我既然沒有用到這個instance的任何東西,那麼我幹嘛要retian它?

我們要注意的一點是,我看到網上有很多人說block引起了例項與block之間的迴圈引用(retain-cycle),並且給出解決方案:不直接使用self而先將self賦值給乙個臨時變數,然後再使用這個臨時變數。

但是,大家注意,我們一定要為這個臨時變數增加__block標記。

Block介紹(二)記憶體管理與其他特性

我們在前一章介紹了block的用法,而正確使用block必須要求正確理解block的記憶體管理問題。這一章,我們只陳述結果而不追尋原因,我們將在下一章深入其原因。我們針對不同情況來討論block的存放位置 以下情況中的block位於堆中 1 2 3 4 5 6 7 8 9 10 11 12 13 1...

Block介紹(二)記憶體管理與其他特徵

我們在前一章介紹了block的用法,而正確使用block必須要求正確理解block的記憶體管理問題。這一章,我們只陳述結果而不追尋原因,我們將在下一章深入其原因。我們針對不同情況來討論block的存放位置 以下情況中的block位於堆中 1 2 3 4 5 6 7 8 9 10 11 12 13 1...

iOS中Block介紹(二)記憶體管理與其他特性

我們針對不同情況來討論block的存放位置 1.棧和堆 以下情況中的block位於堆中 void foo blk在棧裡 blkinheap block copy blk blkinheap在堆裡 void foobar void oblkinheap void oblk copy oblkinhea...