Block學習總結(二)

block

1.block的實質

之前說過其實block的本質就是"帶有自動變量的匿名函數"。block類型的變量與函數指針類似,僅是將 * 號換成了 ^ 符號.那么block的底層實現又是如何。 這里可以通過 clang命令將block語法轉換成底層的c語言。具體的實現: 我們可以打開終端,進入想要轉換的block文件目錄下,執行"clang -rewrite-objc 文件名"指令。至此就能將原文件轉換成.cpp的底層文件了。

void test() {
    void(^blk)() = ^(){
        printf("is block");
    };
    blk();
}

以上的代碼通過clang 指令后,轉換成以下的源碼。(這里指摘取出與block語法相關的代碼)

 struct __block_impl {
 void *isa;
 int Flags;
 int Reserved;
 void *FuncPtr;
 };
 
 static struct __test_block_desc_0 {
 size_t reserved;
 size_t Block_size;
 }
 __test_block_desc_0_DATA = { 0, sizeof(struct __test_block_impl_0)
 
 };
 
 struct __test_block_impl_0 {
 
 struct __block_impl impl;
 struct __test_block_desc_0* Desc;
 
 __test_block_impl_0(void *fp, struct __test_block_desc_0 *desc, int flags=0) {
 impl.isa = &_NSConcreteStackBlock;
 impl.Flags = flags;
 impl.FuncPtr = fp;
 Desc = desc;
 }
 
 };
 
 static void __test_block_func_0(struct __test_block_impl_0 *__cself) {
 
 printf("is block");
 }

 void test() {
 
 void(*blk)() = ((void (*)())&__test_block_impl_0((void *)__test_block_func_0, &__test_block_desc_0_DATA));
 
 ((void (*)(__block_impl *))((__block_impl *)blk)->FuncPtr)((__block_impl *)blk);
 }

首先,看到原來的test 函數轉換成了

//源代碼
void test() {
 //將block函數賦值給變量blk
    void(^blk)() = ^(){
        printf("is block");
    };
 //執行block
    blk();
}

<===>

//轉換后的代碼
void test() {
 
 //將block函數賦值給變量blk
 void(*blk)() = ((void (*)())&__test_block_impl_0((void *)__test_block_func_0, &__test_block_desc_0_DATA));
 
 //執行block
 ((void (*)(__block_impl *))((__block_impl *)blk)->FuncPtr)((__block_impl *)blk);
 }
 
 <===>
 //為了更好的濾清代碼,去掉類型的強轉
 //去除強轉后的代碼
void test() {

 //將block函數賦值給變量blk
 *blk = & __test_block_impl_0(*__test_block_func_0, &__test_block_desc_0_DATA);
 
 //執行block
 blk->FuncPtr(blk);

 }

先看 將block函數賦值給變量blk 這段代碼,它的實質是將 " _ test_block_impl_0類型的實例的指針”賦值給了 “ test_block_impl_0指針類型的變量blk”。 而 _ _test_block_impl_0 是一個結構體。

__test_block_impl_0結構體

為什么這個結構體的名稱叫 __test_block_impl_0,它的命名規則是 “ _ _ _block所在的函數的名字block_impl第幾個block ”。

void test1() {
    void(^blk)() = ^(){
        printf("is block");
    };
    blk();
    
    void(^blk2)() = ^(){
        printf("is block2");
    };
    blk2();
}

則結構體為

__test1_block_impl_0

__test1_block_impl_1

接下來看看 __test_block_impl_0結構體具體的構造

 struct __test_block_impl_0 {
 
 struct __block_impl impl; //成員變量 impl
 struct __test_block_desc_0* Desc; // 成員變量 Desc
 
//結構體的構造方法
 __test_block_impl_0(void *fp, struct __test_block_desc_0 *desc, int flags=0) {
 impl.isa = &_NSConcreteStackBlock;
 impl.Flags = flags;
 impl.FuncPtr = fp;
 Desc = desc;
 }
 
 };

它由2個成員變量和一個構造函數組成. 第一個成員變量impl是一個 __block_impl類型的結構體

 struct __block_impl {
 void *isa; // isa 指針
 int Flags; //標記 Flags
 int Reserved;
 void *FuncPtr; //函數指針 FuncPtr
 };
 

第二個成員變量desc 是一個指向 " _ _test_block_desc_0" 結構體的指針。并且定義了一個名為__test_block_desc_0_DATA 的變量(初始化block時用到)。

 static struct __test_block_desc_0 {
 size_t reserved;
 size_t Block_size; //block大小
 }
 
 __test_block_desc_0_DATA = { 
 0, 
 sizeof(struct __test_block_impl_0) // __test_block_impl_0大小
 };
 

再來看下源代碼中初始化結構體的函數,將參數帶入到結構體構造函數后

& __test_block_impl_0(*__test_block_func_0, &__test_block_desc_0_DATA);

<===>

 __test_block_impl_0(*__test_block_func_0, &__test_block_desc_0_DATA, int flags=0) {
 impl.isa = &_NSConcreteStackBlock;
 impl.Flags = flags;
 impl.FuncPtr = __test_block_func_0;
 Desc = __test_block_desc_0_DATA;
 }

這里還需要解釋一個 _test_block_func_0 的變量,它的本質是一個普通的函數,命名規則 " _ _block所在的函數名block_func第幾個block"。其實它就是原block的執行部分。

 static void __test_block_func_0(struct __test_block_impl_0 *__cself) {
 printf("is block");
 }
 
 <===>
 相當于原block函數的執行部分
 ^(){
        printf("is block");
    };

參數 _ _cself 是block本事,相當于 oc 函數中的self。(在改block中還沒有用到__cself)

block的調用

最后就是執行block,通過上面的分析我們其實就知道了block的調用就是通過函數指針執行block函數。并將block本事作為參數傳入。

 blk->FuncPtr(blk);
 <===>
 blk.impl.FuncPtr = tempFuncptr;
 *tempFuncptr = &__test_block_func_0;
 __test_block_func_0(blk);

2.引用外部變量的實質

當block引用了外部變量的時候,它與沒有引用外部變量的時候,通過clang轉換的源碼來看是有不同的。例如,下面的這段引用了外部變量的block

void testBlock2() {    
    int a = 10;
    int b = 20;
    void(^blk)() = ^(){
        printf("a = %d",a);
    };
    blk();
}

通過 clang指令,它將轉換成了如下的代碼

 <=== 以下代碼與未使用外部變量有區別 ===>
 void testBlock2()
 int a = 10;
 int b = 20;
 void(*blk)() = ((void (*)())&__testBlock2_block_impl_0((void *)__testBlock2_block_func_0, &__testBlock2_block_desc_0_DATA, a));
 ((void (*)(__block_impl *))((__block_impl *)blk)->FuncPtr)((__block_impl *)blk);
 }
 
 struct __testBlock2_block_impl_0 {
 struct __block_impl impl;
 struct __testBlock2_block_desc_0* Desc;
 int a;
 __testBlock2_block_impl_0(void *fp, struct __testBlock2_block_desc_0 *desc, int _a, int flags=0) : a(_a) {
 impl.isa = &_NSConcreteStackBlock;
 impl.Flags = flags;
 impl.FuncPtr = fp;
 Desc = desc;
 }
 };

 static void __testBlock2_block_func_0(struct __testBlock2_block_impl_0 *__cself) {
 int a = __cself->a; // bound by copy
 printf("a = %d",a);
 }
 
<=== 以下代碼與未使用外部變量相同 ===>
 struct __block_impl {
 void *isa;
 int Flags;
 int Reserved;
 void *FuncPtr;
 };
 
 static struct __testBlock2_block_desc_0 {
 size_t reserved;
 size_t Block_size;
 } __testBlock2_block_desc_0_DATA = { 0, sizeof(struct __testBlock2_block_impl_0)};

來對比一下與未使用外部變量時區別的部分,首先是__testBlock2_block_impl_0這個結構體,它多出了一個與引用的外部變量相同類型的成員變量。(這里只添加了block內部使用了的外部變量a這一個成員變量,外部變量b沒有使用,所以沒有添加到結構體中):

 struct __testBlock2_block_impl_0 {
 struct __block_impl impl;
 struct __testBlock2_block_desc_0* Desc;
 int a;  //引用的外部變量 追加成了 __testBlock2_block_impl_0 結構體的成員變量
 };

再來看一下__testBlock2_block_impl_0實例初始化的方法:

 __testBlock2_block_impl_0(void *fp, struct __testBlock2_block_desc_0 *desc, int _a, int flags=0) : a(_a) {
 impl.isa = &_NSConcreteStackBlock;
 impl.Flags = flags;
 impl.FuncPtr = fp;
 Desc = desc;
 }

這個初始化方法與未使用外部變量時的區別是多了一個參數a,而: a(_a)這個語法是c++中結構體中給const類型的變量初始化的意思。即將參數_a賦值給結構體的變量a。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,321評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,559評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,442評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,835評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,581評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,922評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,931評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,096評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,639評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,374評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,591評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,104評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,789評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,196評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,524評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,322評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,554評論 2 379

推薦閱讀更多精彩內容