Block 那些事

Block 那些事

概念

Block是iOS4.0+ 和Mac OS X 10.6+ 引進(jìn)的對C語言的擴(kuò)展该贾,用來實(shí)現(xiàn)匿名函數(shù)的特性
閉包是一個能夠訪問其他函數(shù)內(nèi)部變量的函數(shù).

基本語法

^(returnType)(argsList)
body
}
returnType可以省略握巢,argsList如果沒有可以省略,最簡單的block語法形式如:
^
body;
}
定義一個無參彻亲,返回值為void的block堕担。
如果定義一個block變量介褥,則語法形式為:
returnType (^blockVar)(argsList)
body;
}
例如:
int (^blockVar)(NSString* arg)
return @“Hello World”;
}
上述定義一個 返回值為整數(shù)座掘,帶有一個字符串參數(shù)的blockVar變量。
為了方便使用常常使用typedef給block類型定義一個別名柔滔,如
typedef int (^returnIntBlock)(NSSstring *arg);
returnIntBlock blockVar;
此時returnIntBlock就代表反回值為整數(shù)參數(shù)為字符串的block類型溢陪。blockVar 代表 該block類型的一個變量

基本用法
  1. 定義普通變量睛廊。

     //定義一個返回值為整型形真,帶有兩個整形參數(shù)的block變量
     NSInteger (^returnIntVar)(NSInteger, NSInteger);
     //給block變量賦值
     returnIntVar = ^NSInteger(NSInteger left, NSInteger right){
             return left + right;
         };
    
  2. 定義屬性。

     @property (nonatomic, copy) NSInteger (^returnIntBlock)(NSInteger, NSInteger);
    
  3. block作為方法參數(shù)超全。

     - (void)methodWithBlockArg:(NSInteger (^)(NSInteger, NSInteger))blockArg{
         NSInteger sum ;
         sum =  blockArg(3, 4);
     }
    
  4. block作為方法的返回值咆霜。
    - (void (^)(NSString *arg))methodReturnBlock{

         return ^void(NSString *arg){
             NSLog(@"arg: %@", arg);
         };
     }
    
  5. block作為函數(shù)的返回值邓馒。
    void (^functionReturnBlock(NSString *arg))(NSString *arg){

         return ^void(NSString *arg){
             NSLog(@"arg: %@", arg);
         };
     }
    
block注意事項(xiàng)
  1. 修改引用的外部變量。
    默認(rèn)情況下block內(nèi)部是不能修改應(yīng)用變量的值的蛾坯,若要修改光酣,需在定義外部變量時使用__block關(guān)鍵字修飾。
    __block NSInteger a = 3;
    void(^blockOp)() = ^(){
    a = 5;
    };
    blockOp();

  2. 循環(huán)引用脉课。最常見的情況是對象擁有block挂疆,block內(nèi)部又去訪問對象的屬性,這時下翎,在block外面定義一個弱引用。如下
    __weak typeof(self) weakSelf = self;
    self.returnIntBlock = ^(NSInteger left, NSInteger right){

              weakSelf.sum =    left + right ;
             
             return weakSelf.sum;
         };
    
block原理

為了研究編譯器是如何實(shí)現(xiàn) block 的宝当,我們需要使用 clang视事。clang 提供一個命令,可以將 Objetive-C 的源碼改寫成 c 語言的庆揩,借此可以研究 block 具體的源碼實(shí)現(xiàn)方式俐东。該命令是:

clang -rewrite-objc block.c
我們先新建一個名為 block1.c 的源文件:
include <stdio.h>

int main()
{
        int a = 100;
        void (^block2)(void) = ^{
        printf("%d\n", a);
        };
        block2();
        return 0;
}

然后在命令行中輸入:

clang -rewrite-objc block1.c

如果成功,會在當(dāng)前目錄下生成一個block.cpp的源文件订晌。該文件中的關(guān)鍵代碼如下:
struct __block_impl {
void *isa;
int Flags;
int Reserved;
void FuncPtr;
};
struct __main_block_impl_0 {
struct __block_impl impl;
struct __main_block_desc_0
Desc;
int a;
__main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _a, int flags=0) : a(_a) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
int a = __cself->a; // bound by copy
printf("%d\n", a);
}

static struct __main_block_desc_0 {
size_t reserved;
size_t Block_size;
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};

int main()
{
int a = 100;
void (*block2)(void) = (void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, a);
((void (*)(__block_impl *))((__block_impl *)block2)->FuncPtr)((__block_impl *)block2);
return 0;
}

其中:

  • isa 指針虏辫,所有對象都有該指針,用于實(shí)現(xiàn)對象相關(guān)的功能锈拨。

  • flags砌庄,用于按 bit 位表示一些 block 的附加信息。

  • reserved奕枢,保留變量娄昆。

  • funcPtr,函數(shù)指針缝彬,指向具體的 block 實(shí)現(xiàn)的函數(shù)調(diào)用地址萌焰。

  • desc, 表示該 block 的附加描述信息谷浅,主要是 size 大小扒俯,以及 copy 和 dispose 函數(shù)的指針。

  • a一疯,capture 過來的變量撼玄,block 能夠訪問它外部的局部變量,就是因?yàn)閷⑦@些變量(或變量的地址)復(fù)制到了結(jié)構(gòu)體中墩邀。
    我們修改上面的源碼互纯,在變量前面增加 __block關(guān)鍵字:
    include <stdio.h>

    int main()
    {
    __block int i = 1024;
    void (^block1)(void) = ^{
    printf("%d\n", i);
    i = 1023;
    };
    block1();
    return 0;
    }

生成的關(guān)鍵代碼如下,可以看到磕蒲,差異相當(dāng)大:

    struct __Block_byref_i_0 {
    void *__isa;
    __Block_byref_i_0 *__forwarding;
    int __flags;
    int __size;
    int i;
    };
    
    struct __main_block_impl_0 {
    struct __block_impl impl;
    struct __main_block_desc_0* Desc;
    __Block_byref_i_0 *i; // by ref
    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_i_0 *_i, int flags=0) : i(_i->__forwarding) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
    }
    };
    static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
    __Block_byref_i_0 *i = __cself->i; // bound by ref
    
    printf("%d\n", (i->__forwarding->i));
    (i->__forwarding->i) = 1023;
    }
    
    static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->i, (void*)src->i, 8/*BLOCK_FIELD_IS_BYREF*/);}
    
    static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->i, 8/*BLOCK_FIELD_IS_BYREF*/);}
    
    static struct __main_block_desc_0 {
    size_t reserved;
    size_t Block_size;
    void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*);
    void (*dispose)(struct __main_block_impl_0*);
    } __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0};
    
    int main()
    {
    __attribute__((__blocks__(byref))) __Block_byref_i_0 i = {(void*)0,(__Block_byref_i_0 *)&i, 0, sizeof(__Block_byref_i_0), 1024};
    void (*block1)(void) = (void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, (__Block_byref_i_0 *)&i, 570425344);
    ((void (*)(__block_impl *))((__block_impl *)block1)->FuncPtr)((__block_impl *)block1);
    return 0;
    }

源碼中增加一個名為 __Block_byref_i_0 的結(jié)構(gòu)體留潦,用來保存我們要 capture 并且修改的變量 a只盹。
main_block_impl_0 中引用的是 Block_byref_i_0 的結(jié)構(gòu)體指針,這樣就可以達(dá)到修改外部變量的作用兔院。
對于 block 外的變量引用殖卑,block 默認(rèn)是將其復(fù)制到其數(shù)據(jù)結(jié)構(gòu)中來實(shí)現(xiàn)訪問的。


對于用 __block 修飾的外部變量引用坊萝,block 是復(fù)制其引用地址來實(shí)現(xiàn)訪問的孵稽。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市十偶,隨后出現(xiàn)的幾起案子菩鲜,更是在濱河造成了極大的恐慌,老刑警劉巖惦积,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件接校,死亡現(xiàn)場離奇詭異,居然都是意外死亡狮崩,警方通過查閱死者的電腦和手機(jī)蛛勉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來睦柴,“玉大人诽凌,你說我怎么就攤上這事√沟校” “怎么了侣诵?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長狱窘。 經(jīng)常有香客問我窝趣,道長,這世上最難降的妖魔是什么训柴? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任哑舒,我火速辦了婚禮,結(jié)果婚禮上幻馁,老公的妹妹穿的比我還像新娘洗鸵。我一直安慰自己,他們只是感情好仗嗦,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布膘滨。 她就那樣靜靜地躺著,像睡著了一般稀拐。 火紅的嫁衣襯著肌膚如雪火邓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天,我揣著相機(jī)與錄音铲咨,去河邊找鬼躲胳。 笑死,一個胖子當(dāng)著我的面吹牛纤勒,可吹牛的內(nèi)容都是我干的坯苹。 我是一名探鬼主播,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼摇天,長吁一口氣:“原來是場噩夢啊……” “哼粹湃!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起泉坐,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤为鳄,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后腕让,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體孤钦,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年记某,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片构捡。...
    茶點(diǎn)故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡液南,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出勾徽,到底是詐尸還是另有隱情滑凉,我是刑警寧澤,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布喘帚,位于F島的核電站畅姊,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏吹由。R本人自食惡果不足惜若未,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望倾鲫。 院中可真熱鬧粗合,春花似錦、人聲如沸乌昔。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽磕道。三九已至供屉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背伶丐。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工悼做, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人撵割。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓贿堰,卻偏偏與公主長得像,于是被迫代替她去往敵國和親啡彬。 傳聞我的和親對象是個殘疾皇子羹与,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 前言 Blocks是C語言的擴(kuò)充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了這...
    小人不才閱讀 3,768評論 0 23
  • 摘要block是2010年WWDC蘋果為Objective-C提供的一個新特性庶灿,它為我們開發(fā)提供了便利纵搁,比如GCD...
    西門吹雪123閱讀 916評論 0 4
  • Blocks Blocks Blocks 是帶有局部變量的匿名函數(shù) 截取自動變量值 int main(){ ...
    南京小伙閱讀 929評論 1 3
  • Block基礎(chǔ)回顧 1.什么是Block? 帶有局部變量的匿名函數(shù)(名字不重要往踢,知道怎么用就行)腾誉,差不多就與C語言...
    Bugfix閱讀 6,766評論 5 61
  • iOS代碼塊Block 概述 代碼塊Block是蘋果在iOS4開始引入的對C語言的擴(kuò)展,用來實(shí)現(xiàn)匿名函數(shù)的特性,B...
    smile刺客閱讀 2,349評論 2 26