Block

Block基本語法

typedef聲明,作Method參數(shù)使用
typedef void(^typedefBlock)(int age,NSString *str);
- (void)blockTypedefMethodWithBlock:(typedefBlock)callBack{
    callBack(18,@"is my age");
}
屬性聲明,使用copy修飾符
@property (nonatomic, copy) typedefBlock myBlock;
基本語法
/**
 block基本語法
 return_type (^blockName)(var_type) = ^return_type (var_type varName) { // ... };
 blockName(var);
 */
- (void)blockBasicGrammar{
    //有參無返回值
    void(^block1)(int index,NSString *name) = ^void(int index,NSString *name){
        NSLog(@"index = %d name = %@",index,name);
    };
    block1(3,@"kelly");
}

聲明Block屬性使用copy修飾詞

Block三種類型
  • 全局塊(_NSConcreteGlobalBlock)
  • 棧塊(_NSConcreteStackBlock)
  • 堆塊(_NSConcreteMallocBlock)

存儲(chǔ)區(qū)域
1. 全局塊存儲(chǔ)在靜態(tài)區(qū)(也叫全局區(qū)),相當(dāng)于OC中的單例;
2. 棧塊存儲(chǔ)在棧區(qū),超出作用域馬上被銷毀;
3. 堆塊存儲(chǔ)在堆區(qū),是一個(gè)帶引用計(jì)數(shù)的對(duì)象,需自行管理內(nèi)存。


判斷block存儲(chǔ)位置

  1. block不訪問外界變量(包括棧中和堆中的變量):block既不在棧中也不在堆中,此時(shí)為全局塊,ARC和MRC都如此;
  2. block訪問外界變量:
    ?MRC:默認(rèn)存儲(chǔ)在棧區(qū);
    ?ARC:默認(rèn)存儲(chǔ)在堆中,實(shí)際上是先放在棧區(qū),在ARC情況下自動(dòng)拷貝到堆區(qū),自動(dòng)釋放。

block使用copy修飾符的作用:將block從棧區(qū)拷貝到堆區(qū)。
?原因:保存block狀態(tài),延長生命周期。因?yàn)閎lock存儲(chǔ)在棧,其所屬變量作用域結(jié)束,該block就被釋放,__block變量也同時(shí)被釋放掉,所以需要copy到堆中。

注意:不同類型block使用copy效果也不一樣,如下圖:

Block拷貝.png


__block的使用(改變block的存儲(chǔ)區(qū))

例子1,不使用__block

- (void)testMethod {
    int anInteger = 42;
 
    void (^testBlock)(void) = ^{
        NSLog(@"Integer is : %i", anInteger);
    };
 
    anInteger = 50;
 
    testBlock();
}

輸出結(jié)果:Integer is : 42

例子2,使用__block

- (void)testMethod {
    __block int anInteger = 42;
 
    void (^testBlock)(void) = ^{
        NSLog(@"Integer is : %i", anInteger);
    };
 
    anInteger = 50;
 
    testBlock();
}

輸出結(jié)果:Integer is : 50

分析
?例子1:block把a(bǔ)ninteger變量復(fù)制為自己私有的const變量,也就是說block會(huì)捕獲棧上的變量或指針,將其復(fù)制為自己私有的const變量。在進(jìn)行aninterger=50的操作時(shí),block已經(jīng)將其復(fù)制為自己的私有變量,所以修改aninterger的值不會(huì)對(duì)block里面的值造成影響。
?例子2:aninteger是一個(gè)局部變量,存儲(chǔ)在棧區(qū)。給aninteger加入__block修飾符的作用是只要觀察到該變量被block持有,就將該變量在棧中的內(nèi)存地址存在到堆中,此時(shí)不管block外部還是內(nèi)部aninteger的內(nèi)存地址都是一樣的,進(jìn)而可以修復(fù)aninteger變量的值。
??圖例

__block.png

block循環(huán)引用

循環(huán)引用例子

@property (nonatomic, copy) void (^block)(void);
- (void)configureBlock {
    self.block = ^{
        [self doSomething];    
    };
}

?在上面代碼中聲明了一個(gè)屬性block,所以self對(duì)block有一個(gè)強(qiáng)引用。而block內(nèi)部又對(duì)self進(jìn)行一次強(qiáng)引用,這樣就形成了一個(gè)封閉的環(huán),也就是強(qiáng)引用循環(huán)。引用關(guān)系如圖:

強(qiáng)引用循環(huán).png

?在這種情況下,由于相互引用,內(nèi)存不能進(jìn)行釋放,造成內(nèi)存泄漏。解決方案:__weak

- (void)configureBlock {
    __weak typeof(self) weakSelf = self;
    self.block = ^{
        [weakSelf doSomething];
    };
}

?使用__weak之后,block對(duì)self就由強(qiáng)引用變?yōu)槿跻茫@樣在屬性所指的對(duì)象被釋放時(shí),屬性值也會(huì)被釋放,打破了block捕獲作用域帶來的循環(huán)引用。
?需要注意的是:使用系統(tǒng)的block API不需要考慮循環(huán)引用問題,因?yàn)橹挥衎lock對(duì)self進(jìn)行了一次強(qiáng)引用,屬于單向強(qiáng)引用,沒有形成循環(huán)引用。


weak與strong區(qū)別

- (void)configureBlock {
    __weak typeof(self) weakSelf = self;
    self.block = ^{
        [weakSelf doSomething]; // weakSelf != nil
        // preemption(搶占) weakSelf turned nil
        [weakSelf doAnotherThing];
    };
}

上面這段代碼看上去很正常,但是在并發(fā)執(zhí)行的時(shí)候,block的執(zhí)行時(shí)可以搶占的,而且對(duì)weakSelf指針的調(diào)用時(shí)序不同可以導(dǎo)致不同的結(jié)果,比如在一個(gè)特定時(shí)序下weakSelf可能變成nil,這個(gè)時(shí)候執(zhí)行doAnotherThing就會(huì)崩潰。為避免這樣的問題,使用__strong進(jìn)行規(guī)避,代碼修改:

- (void)configureBlock {
    __weak typeof(self) weakSelf = self;
    self.block = ^{
        __strong typeof(weakSelf) strongSelf = weakSelf;
        [strongSelf doSomething]; // strongSelf != nil
        // 在搶占的時(shí)候,strongSelf還是非nil的。
        [strongSelf doAnotherThing];
    };
}
總結(jié)
  • block不作為property的時(shí)候,可以在block里面直接使用self,比如UIView的animation動(dòng)畫block;
  • block被聲明property的時(shí)候,需要使用__weak弱引用,并解決循環(huán)引用的問題;
  • 當(dāng)和并發(fā)執(zhí)行相關(guān)、涉及異步的服務(wù)的時(shí)候,block可以在之后被執(zhí)行,并且不會(huì)發(fā)生關(guān)于self是否存在的問題。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,563評(píng)論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,694評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,672評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,965評(píng)論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,690評(píng)論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 56,019評(píng)論 1 329
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,013評(píng)論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,188評(píng)論 0 290
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,718評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,438評(píng)論 3 360
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,667評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,149評(píng)論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,845評(píng)論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,252評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,590評(píng)論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,384評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,635評(píng)論 2 380