從0開始探究blcok底層

關(guān)于這方面的內(nèi)容其實滿天飛,但每篇都幾乎是一樣子的,可能大多數(shù)人都覺得自己講的特別清楚抖誉,但是其實很多內(nèi)容都讓讀者覺得作者說的云里霧里估脆,不知所云钦奋。今天嘗試從0開始一直到block講的清清楚楚為止。
我們使用

clang -rewrite-objc -fobjc-arc -stdlib=libc++ -mmacosx-version-min=10.7 -fobjc-runtime=macosx-10.7 -Wno-deprecated-declarations main.m

命令重寫以下代碼,看一下各種類型的變量變成C++后是什么樣子旁蔼。

1锨苏、int 類型變量

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int a = 1;
    }
    return 0;
}

重寫后生成的C++代碼多大9萬多行,不過沒關(guān)系棺聊,我們把文件拉到最后伞租,直接看這里

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        int a = 1;
    }
    return 0;
}

這幾萬行代碼的上面都是 #import <Foundation/Foundation.h>
首先是 @autoreleasepool 語法糖被重寫成了 __AtAutoreleasePool __autoreleasepool 這里暫時先不探究 @autoreleasepool的問題,先看一下變量的情況限佩,可以看到下面并沒有發(fā)生什么變化葵诈,仍然是 int a = 10;符合預(yù)期因為這本來就是 C 語法嘛裸弦。

2、__block 修飾的 int 類型變量

這里就有點意思了作喘,我們聲明一個帶__block修飾的變量如下

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        __block int a = 1;
    }
    return 0;
}

同樣執(zhí)行上述命令編譯成C++代碼如下:

struct __Block_byref_a_0 { // __block修飾的int變量生成的結(jié)構(gòu)體
  void *__isa;
__Block_byref_a_0 *__forwarding; // 該結(jié)構(gòu)體指向的實際值指針
 int __flags; // 標(biāo)識位
 int __size; // 該結(jié)構(gòu)體的內(nèi)存大小
 int a; // 保存的實際值
};
int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        __attribute__((__blocks__(byref))) __Block_byref_a_0 a = {(void*)0,(__Block_byref_a_0 *)&a, 0, sizeof(__Block_byref_a_0), 1};
    }
    return 0;
}

可以看到__block int a = 1被編譯成了這么多東西理疙。參考這里
https://gist.github.com/mjhsieh/668374
可以知道 __block 其實是 __attribute__((__blocks__(byref))) 的宏,那么我們聲明一個帶__block的變量實際上就是聲明了一個上述類型的結(jié)構(gòu)體并且聲明了一個該結(jié)構(gòu)體的實例變量泞坦。

3窖贤、沒有參數(shù)沒有返回值的block 的聲明

我們聲明一個block如下

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(void) = ^() {
            
        };
    }
    return 0;
}

編譯成C++代碼如下

struct __block_impl { // block 的實現(xiàn)
    void *isa;
    int Flags;
    int Reserved;
    void *FuncPtr; // 函數(shù)指針
};

struct __main_block_impl_0 { // main 函數(shù)中 block 的實現(xiàn)
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp; // 該 block 所指向的函數(shù)指針,真正調(diào)用block的時候?qū)嶋H上是執(zhí)行的該函數(shù)
    Desc = desc;// block 的描述信息
  }
};
// 實際函數(shù),由于這個block并不會執(zhí)行任何代碼贰锁,所以該函數(shù)里并沒有任何東西赃梧。該函數(shù)的參數(shù)只有一個,即傳入該結(jié)構(gòu)體的實例豌熄。
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {


        }

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 argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
      //這里執(zhí)行的是該結(jié)構(gòu)體的構(gòu)造函數(shù)授嘀,將上述函數(shù)和block的描述結(jié)構(gòu)體傳入,構(gòu)造一個block變量锣险。
        void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
    }
    return 0;
}

可以看到最后生成了三個結(jié)構(gòu)體和一個靜態(tài)函數(shù)蹄皱。其中最主要的就是__main_block_impl_0這個結(jié)構(gòu)體
,里面保存著該block實際調(diào)用的函數(shù)的指針和該block的描述信息芯肤。描述信息主要記錄著該block的大小巷折。

4、有參數(shù)沒有返回值的block 的聲明和執(zhí)行

我們試著將如下代碼轉(zhuǎn)成C++代碼來看

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        void(^block)(int) = ^(int a) {
            NSLog(@"%d", a);
        };
        block(2);
    }
    return 0;
}

C++ 代碼如下

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        void(*block)(int) = ((void (*)(int))&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
        ((void (*)(__block_impl *, int))((__block_impl *)block)->FuncPtr)((__block_impl *)block, 2);
    }
    return 0;
}

可以看到執(zhí)行block的實際含義其實是調(diào)用該block中保存的函數(shù)指針纷妆,然后將該block以及block需要的參數(shù)傳入即可盔几,即執(zhí)行了該函數(shù)。

5掩幢、block 引用普通變量

OC代碼如下

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        int b = 1;
        void(^block)(void) = ^{
            NSLog(@"%d", b);
        };
        block();
    }
    return 0;
}

編譯成的C++代碼如下

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int b;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _b, int flags=0) : b(_b) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  int b = __cself->b; // bound by copy

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_p0_xw8yd5qj1rd_0pyjtn4g0t7h0000gn_T_main_ec2b7f_mi_0, b);
        }

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 argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        int b = 1;
        void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, b));
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
    }
    return 0;
}

可以看到block生成的結(jié)構(gòu)體中有了block里使用的變量的值逊拍,即block拷貝了該變量到自己的結(jié)構(gòu)體里面,這也是當(dāng)block聲明之后再修改變量的值,block里的變量不會變化的原因际邻。因為要執(zhí)行的函數(shù)已經(jīng)確定了芯丧,相應(yīng)的值也是確定的,外面再修改已經(jīng)沒有用了世曾,因為block對于普通變量執(zhí)行的是值拷貝的操作缨恒。

6、__block 修飾的變量在block中使用的場景

OC代碼如下

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        __block int b = 1;
        void(^block)(void) = ^{
            NSLog(@"%d", b);
        };
        b = 2;
        block();
    }
    return 0;
}

編譯后生成的C++代碼如下

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
        __attribute__((__blocks__(byref))) __Block_byref_b_0 b = {(void*)0,(__Block_byref_b_0 *)&b, 0, sizeof(__Block_byref_b_0), 1};
        void(*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, (__Block_byref_b_0 *)&b, 570425344));
        (b.__forwarding->b) = 2;
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
    }
    return 0;
}

可以看到傳入block執(zhí)行函數(shù)的是__block int b 生成的結(jié)構(gòu)體轮听,改變b的值其實是改變的該結(jié)構(gòu)體中的b的值骗露,當(dāng)執(zhí)行該函數(shù)的時候傳入的參數(shù)是該結(jié)構(gòu)體的值,所以block執(zhí)行的時候可以拿到最新的值血巍。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末萧锉,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子述寡,更是在濱河造成了極大的恐慌柿隙,老刑警劉巖叶洞,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異禀崖,居然都是意外死亡衩辟,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進(jìn)店門波附,熙熙樓的掌柜王于貴愁眉苦臉地迎上來艺晴,“玉大人,你說我怎么就攤上這事掸屡〔萍ⅲ” “怎么了?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵折晦,是天一觀的道長。 經(jīng)常有香客問我沾瓦,道長满着,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任贯莺,我火速辦了婚禮风喇,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘缕探。我一直安慰自己魂莫,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布爹耗。 她就那樣靜靜地躺著耙考,像睡著了一般。 火紅的嫁衣襯著肌膚如雪潭兽。 梳的紋絲不亂的頭發(fā)上倦始,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天,我揣著相機與錄音山卦,去河邊找鬼鞋邑。 笑死,一個胖子當(dāng)著我的面吹牛账蓉,可吹牛的內(nèi)容都是我干的枚碗。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼铸本,長吁一口氣:“原來是場噩夢啊……” “哼肮雨!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起归敬,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤酷含,失蹤者是張志新(化名)和其女友劉穎鄙早,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體椅亚,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡限番,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了呀舔。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片弥虐。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖媚赖,靈堂內(nèi)的尸體忽然破棺而出霜瘪,到底是詐尸還是另有隱情,我是刑警寧澤惧磺,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布颖对,位于F島的核電站,受9級特大地震影響磨隘,放射性物質(zhì)發(fā)生泄漏缤底。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一番捂、第九天 我趴在偏房一處隱蔽的房頂上張望个唧。 院中可真熱鬧,春花似錦设预、人聲如沸徙歼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽魄梯。三九已至,卻和暖如春耕魄,著一層夾襖步出監(jiān)牢的瞬間画恰,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工吸奴, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留允扇,地道東北人。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓则奥,卻偏偏與公主長得像考润,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子读处,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,786評論 2 345

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