看了看,OC中的Block機制還是跟c中的函數指針很相像。
以下內容爲轉載內容,僅供學習參考
一:基本定義
/*初步上式block定義的一些理解和解釋,接下來會詳解:
*block名爲myBlock,結合C的函數指針,myBlock爲block體的指針,指向block體的入口地址
*int result = myBlock(5) <==> ^(int num){return num*num}(5)//將5傳給num
*回調時可以將myBlock作爲參數傳入,也可以直接傳入block體^(int num){…};//
*整個block體作爲參數傳入時,往往沒有參數,只是進行延遲運算作用,因爲定義了block內容
如果是暫時左值不需要的話,block是不執行的。
例:[request setFailedBlock:^{NSError *error = [request error]; };];
//如果不執行setFailedBlock方法時,後面的block是不執行的,只是靜靜地待在堆中,原理接下來會說到。
*/
二:block原理
首先介紹基本的block,將OC語言block改寫成C語言的block
具體步驟:
我們通過IMP定義可以看出:
在main()函數中,block的返回值爲(void(*)()),其實就是一個函數指針,所在在這種情況下,block就可以看做是匿名函數的函數指針來對待。
三:關鍵字__block
主要作用:
1.block對外部變量是隻讀的,要變成可讀可寫,就需要加上__block。 加上這個關鍵字變量在block中就是深拷貝,不加就是淺拷貝。
2.將棧中的block複製到堆上一份,從而避免了循環引用這個情況
將帶有__block關鍵字的block進行OC->C語言轉換:
四:存儲類型
block的存儲形態有三種:_NSConcretStackBlock(棧)、_NSConcretGlobalBlock(全局)、_NSConcretMallocBlock(堆)
要點一:當block在函數內部,且定義的時候就使用了函數內部的變量,那麼這個 block是存儲在棧上的。
要點二:當block定義在函數體外面,或者定義在函數體內部且當時函數執行的時候,block體中並沒有需要使用函數內部的局部變量時,也就是block在函數執行的時候只是靜靜地待在一邊定義了一下而不使用函數體的內容,那麼block將會被編譯器存儲爲全局block。
要點三:全局block存儲在堆中,對全局block使用copy操作會返回原函數指針;而對棧中的block使用copy操作,會產生兩個不同的block地址,也就是兩個匿名函數的入口地址。
要點四:ARC機制優化會將stack的block,轉爲heap的block進行調用。