Q:什么是Block?
A:Block是將函數(shù)及其執(zhí)行上下文封裝起來的對(duì)象
block的三種存儲(chǔ)狀態(tài):
NSGlobalBlock 在靜態(tài)存儲(chǔ)區(qū),生存周期長(zhǎng),對(duì)其release或者copy retain沒有用。是計(jì)算機(jī)管理的空間,程序退出就釋放空間。
NSStackBlock 在棧上,對(duì)其進(jìn)行retain 或者release無效,棧上空間是計(jì)算機(jī)自動(dòng)釋放的,copy后會(huì)拷貝到堆空間,下來就是nsmallocBlock。
NSMallocBlock 是堆上空間,對(duì)其retain,release,copy都有用,但是其引用計(jì)數(shù)器都是1,打印不會(huì)變。
Block 關(guān)鍵字使用
使用弱引用會(huì)帶來另一個(gè)問題,weakSelf 有可能會(huì)為 nil,如果多次調(diào)用 weakSelf 的方法,有可能在 block 執(zhí)行過程中 weakSelf 變?yōu)?nil。因此需要在 block 中將 weakSelf “強(qiáng)化“
__weak __typeof__(self) weakSelf = self;
NSBlockOperation *op = [[[NSBlockOperation alloc] init] autorelease];
[ op addExecutionBlock:^ {
__strong __typeof__(self) strongSelf = weakSelf;
[strongSelf doSomething];
[strongSelf doMoreThing];
} ];
[someOperationQueue addOperation:op];
__strong 這一句在執(zhí)行的時(shí)候,如果 WeakSelf 還沒有變成 nil,那么就會(huì) retain self,讓 self 在 block 執(zhí)行期間不會(huì)變?yōu)?nil。這樣上面的 doSomething 和 doMoreThing 要么全執(zhí)行成功,要么全失敗,不會(huì)出現(xiàn)一個(gè)成功一個(gè)失敗,即執(zhí)行到中間 self 變成 nil 的情況。
Block與外界變量
(1)默認(rèn)情況
對(duì)于 block 外的變量引用,block 默認(rèn)是將其復(fù)制到其數(shù)據(jù)結(jié)構(gòu)中來實(shí)現(xiàn)訪問的。也就是說block的自動(dòng)變量截獲只針對(duì)block內(nèi)部使用的自動(dòng)變量, 不使用則不截獲, 因?yàn)榻孬@的自動(dòng)變量會(huì)存儲(chǔ)于block的結(jié)構(gòu)體內(nèi)部, 會(huì)導(dǎo)致block體積變大。
block只能訪問不能修改局部變量的值。
int age = 10;
myBlock block = ^{
NSLog(@"age = %d", age);
};
age = 18;
block(); //輸出 10
(2) __block 修飾的外部變量
對(duì)于用 __block 修飾的外部變量引用,block 是復(fù)制其引用地址來實(shí)現(xiàn)訪問的
__block int age = 10;
myBlock block = ^{
NSLog(@"age = %d", age);
};
age = 18;
block(); //輸出18
2、__block 修飾的外部變量的值就可以被block修改
我們使用 clang 將 OC 代碼轉(zhuǎn)換為 C++ 文件:
clang -rewrite-objc 源代碼文件名
__block int val = 10;
轉(zhuǎn)換成
__Block_byref_val_0 val = {
0,
&val,
0,
sizeof(__Block_byref_val_0),
10
};
會(huì)發(fā)現(xiàn)一個(gè)局部變量加上__block修飾符后竟然跟block一樣變成了一個(gè)__Block_byref_val_0結(jié)構(gòu)體類型的自動(dòng)變量實(shí)例。
此時(shí)我們?cè)赽lock內(nèi)部訪問val變量則需要通過一個(gè)叫__forwarding的成員變量來間接訪問val變量。
block的循環(huán)引用
參考鏈接
如下代碼不需要使用__block,因?yàn)槭菍?duì)數(shù)組的操作而不是數(shù)組的賦值。淺談Block 尾部有Block注意事項(xiàng)題目
NSMutableArray *arrM = [NSMutableArray array];
void (^testBlock)(void) = ^{
[arrM addObject:@"addObj"];
};
testBlock();