iOS 之 Block 原理

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的使用

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();
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容