這篇博客將系統(tǒng)整理一下Block相關的知識
首先,先思考一個問題Block能做些什么?
- Block內部能夠讀取外部局部變量的值
- 經過
__block
修飾符修飾的變量,Block內部不僅能夠讀取它的值,而且可以改變這個變量的值 - Block可以用來作為控制器之間的一個通信
- Block可以作為方法的參數(shù)
進一步深挖block的原理
為什么不加__block
修飾符就不能改變Block外部的局部變量?
對于block外的變量引用,block默認是將其復制到其數(shù)據(jù)結構中實現(xiàn)訪問的,對于用__block
修飾的外部變量引用,block是復制其引用地址來實現(xiàn)訪問的,所以加上__block
就能改變block外部的變量
Block如何作為控制器之間的一個通信?
舉例說明:
- 頁面B的.h文件中聲明一個block變量
// 方法一
typedef void(^blo)(NSString *str);
@property (nonatomic, copy) blo block;
// 方法二
@property (nonatomic, copy) void (^blo)(NSString *str);
- 頁面A中創(chuàng)建B頁面,并調用Block變量
// 方法一
B *b = [[B alloc] init];
__weak A *weakself = self;
b.block = ^(NSString *str) {
weakself.str = str;
}
// 方法二
B *b = [[B alloc] init];
__weak A *weakself = self;
b.blo = ^(NSString *str) {
weakself.str = str;
}
- 在頁面B中調用Block變量
self.block("string");
為何A頁面要用__weak修飾?
使用__weak修飾符是為了避免引起循環(huán)引用。如果在Block中使用了__strong
修飾符修飾的對象,那么當Block從棧復制到堆時,改對象為Block所有,這就容易引起循環(huán)引用,從而發(fā)生內存泄漏
如何使用Block作為方法的參數(shù)
- (void)blockAsParameter:(void (^)())block {
block(); // 執(zhí)行block內部的代碼
}
[self blockAsParameter:^{
NSLog(@"");
}];
block的三種類型
- 不管在ARC還是MRC環(huán)境下,block內部如果沒有訪問外部變量,這個block就是全局block
_NSConcreteGlobalBlock
,存儲在內存中的代碼區(qū) - 在MRC下,block內部如果訪問外部變量,這個block就是棧block
_NSConcreteStackBlock
,存儲在內存中的棧上,當函數(shù)返回時會被銷毀 - 在MRC下,block內部訪問外部變量,同時對該block做一次copy操作,這個block就是堆block
_NSConcreteMallocBlock
,存儲在內存的堆上,當引用計數(shù)為 0 時會被銷毀 - 在ARC下,block內部如果訪問了外部變量,這個block就是堆block,因為在ARC下,默認對block做一次copy操作
總結一些block使用的注意點
1.默認情況下,block內部可以訪問外部變量,但是不能修改外部變量,如果想要修改外部變量,可以給外部變量加上__block
關鍵字,block是復制其引用地址來實現(xiàn)訪問的
- block作為屬性應該用copy修飾,如果我們使用weak、assign來修飾block屬性,block訪問外部變量時類型就是棧block,而保存在棧中的block在其所在的函數(shù)、方法返回時,該block就會被銷毀,在其他方法內部調用改block時,就會引發(fā)野指針錯誤。
參考文獻
黑幕背后的 __block 修飾符--分析了為什么不加__block修飾符就不能改變Block外部的局部變量
歡迎訪問我的個人博客