BLOCK
block的描述: 他是類似函數指針的一個代碼塊的內聯封裝, 他可以將一個函數體作為對象傳遞
block的本質: 對象
block的效率: 高于函數調用
聲明一個block
? ? ? ? ? ?返回值 (^名稱)(以”逗號”分割的參數), 參數部分可以不寫參數名只寫類型
范例:
? ? ? ? ? ? ? void ?(^myFirstBlock)( int a, NSString *s, UIButton *button )
? ? ? ? ? ? BOOL (^mySecondBlock)(void)
? ? NSString * (^myThirdBlock)(CGFloat, float)
定義一個block
? ? ? ? ?^可省略的返回值(以”逗號”分割的參數){代碼塊}
范例:
? ? ? ? ^void(int a){NSLog(@"一個int參數, 無返回值的block"); }?
? ? ? ? ^{ NSLog(@"一個無參數,無返回值的block"); }? ? ? ? ? ? ? ??
? ? ? ? ^int(int a, int b){ NSLog( @"一個兩個int參數, 返回兩個數相加的block, 結果%d", a + b ); return a + b; }
block作為函數參數
? ? ?? ? block作為參數, 參數名后綴, 類型格式為: ( 返回值(^)(參數) )block名?
范例:
- (void)test:(int(^)(int))paramBlock
{
? ? ? ? NSLog( @"打印block返回結果:%d", paramBlock(10));
}
?block作為參數的函數調用:
? ? ? ? ?將block定義部分或將block對象傳入函數調用的參數部分
范例:
[self test:^(int a){
? ? ? ? return a * a;
}];
__block:
block只能只讀的訪問局部變量, 若需要修改局部變量, 需要通過__block修飾該變量, 該修飾符會使block自動復制該對象的指針,即指針拷貝, 當對該對象執行寫操作時, 其實就是操作其指針指向的對象進行操作.
若不加該修飾符, block僅自動復制需要訪問的局部變量的值副本(值拷貝), 因此只能進行只讀操作.
__weak:
使用block特別要注意的一點就是防止循環引用的發生, block作為一個對象被持有者持有,若block內部也持有該持有者將會導致循環引用,導致相互持有, 永遠無法銷毀彼此的內存問題, 若不能規避該問題, 則需要通過使用__weak修飾 block要訪問的持有者變量, 防止循環引用的發生. 尤其是block所在的self.
__strong
該修飾符在block內的主要作用是防止__weak修飾的指針在block運行期間被外部銷毀, 導致block無法順利完成原定功能. 所以當訪問的外部變量用__weak修飾時, block內部最好再有一個__strong修飾的同類型指針指向__weak指針, 這樣就可以確保, 在block內部的這個指針可以運行完成. 隨著block運行完成, __strong修飾的指針也將自動銷毀, 進而__weak指針也會自動置nil.
block外部變量訪問
block可以訪問和修改全局變量
block只能只讀的訪問局部變量(值傳遞, block內部復制了一個副本), 若加了__block修飾, 就改為引用傳遞,可修改局部變量
block底層淺談
block本身包含一個結構體
struct __block_impl
{
void *isa;
int Flags;
int Reserved;
void *FuncPtr;
}
block分為三種類型, __block_impl的isa指針負責標記block類型
_NSConcreteGlobalBlock: 全局靜態block, 當block內不訪問任何外部變量, 或者訪問的是全局作用域,成員變量的, 屬于該類型, 內存管理方式和函數一致
_NSConcreteStackBlock 棧block, 由系統管理內存, 函數返回時銷毀, 訪問的外部變量由block從棧上copy到堆上, 保證了作用域結束后, block仍然可以訪問
_NSConcreteMallocBlock 堆block, 由開發者控制, 引用計數為0時銷毀
block幾種自身copy的情況, 若觸發一下情況, block將自動拷貝自身到堆中, 確保自身作用域內的變量的生命周期更長久
1.顯式執行copy, [block copy];
2.block作為返回值的時候
3.block被賦值給__strong類型的對象或者block的內部變量時
4.block作為參數傳入帶有usingBlock的CocoaFramework方法
5. block作為參數傳入GCD的API時
其他:
__typeof, __typeof__, typeof 這三個關鍵字實現了一個相同的功能, 獲取參數的類型名稱并返回
UIViewController *vc;
__typeof(vc) vc2;
這樣就可以在不知道vc指針類型的情況下, 再聲明一個同類型的vc2