iOS Block學(xué)習(xí)筆記(三) -- Block的本質(zhì)

在網(wǎng)上查到,使用clang/llvm可以將代碼轉(zhuǎn)化成我們可以閱讀的版本C++版本, 本文使用Clang

int main() {
  void (^blk)(void) = ^{printf("Block\n");};
  blk();
  return 0;
}

在Terminal中輸入cc -rewrite-objc 'main.m' 轉(zhuǎn)化以后, 會生成main.cpp文件, 是C++的代碼如下:

//1. __block_impl 結(jié)構(gòu)體定義, 有以下成員變量: `isa`, `Flags`, `Reserved`, `FuncPtr`
struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};

//2. __main_block_impl_0 結(jié)構(gòu)體定義, 新增成員變量`Desc`用于描述Block的特點(diǎn), 以及一個構(gòu)造函數(shù)
struct __main_block_impl_0 {
  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;
    Desc = desc;
  }
};

//3. __main_block_desc_0, 用于描述特定Block的大小信息.
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)};

//4. __main_block_func_0, 特定Block實(shí)際的函數(shù)實(shí)現(xiàn).
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
    printf("Blocks\n");
}

//5. 實(shí)際完成
int main(int argc, const char * argv[]) {
    void (*blk)() = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));

    ((void (*)(__block_impl *))((__block_impl *)blk)->FuncPtr)((__block_impl *)blk);
    return 0;
}
  1. 定義__block_impl結(jié)構(gòu)體, 用面向?qū)ο蟮乃季S,可以理解成Block類型的基類(同OC中的NSObject一樣). 其中isaClass中的isa指針一樣.(具體可以參考Runtime).void *FuncPtr表示這個Block實(shí)際被調(diào)用時(shí)實(shí)際執(zhí)行函數(shù)的函數(shù)指針.
  2. 定義__main_block_impl_0, 實(shí)際這個結(jié)構(gòu)體才是main函數(shù)中我們定義的Block.這個結(jié)構(gòu)體中, 有一個基類__block_impl, 以及關(guān)于本Block類型的的描述struct __main_block_desc_0* Desc, 還有這個結(jié)構(gòu)體的構(gòu)造函數(shù).
  3. 定義__main_block_desc_0結(jié)構(gòu)體,并且創(chuàng)建一個Block相關(guān)的描述結(jié)構(gòu)體,主要是Block占用的大小.
  4. __main_block_func_0表示我們創(chuàng)建的Block變量實(shí)際會執(zhí)行的函數(shù), 也就是void *FuncPtr函數(shù)指針指向的地方.
  5. 我們可以將main函數(shù)改寫一下:
typedef void (*func_t)(void);//重命名函數(shù)指針

int main(int argc, const char * argv[]) {
    //1. 創(chuàng)建Block真實(shí)的變量, 構(gòu)造函數(shù)里面參數(shù)是底層會調(diào)用的靜態(tài)函數(shù)指針, 以及該Block的描述信息結(jié)構(gòu)體變量.
    struct __main_block_impl_0 tmp = __main_block_impl_0(__main_block_func_0, &__main_block_desc_0_DATA));

    //2. 獲取Block變量的指針
    struct __main_block_impl_0 *blk_t = &tmp;

    //3. 獲取實(shí)際要執(zhí)行函數(shù)的函數(shù)指針
    func_t funcPtr = (func_t)(blk_t->impl.FuncPtr);

    //4. 用函數(shù)指針執(zhí)行Block中的函數(shù)
    (*funcPtr)(blk_t);
    return 0;
}

通過Clang編譯器的轉(zhuǎn)化, 我們可以清晰的看清Block的本質(zhì)是一個結(jié)構(gòu)體, 包含很多重要的信息.

我們知道runtime中關(guān)于OC的Class有如下信息:

typedef struct objc_class *Class;

struct objc_class {
  Class isa  OBJC_ISA_AVAILABILITY;
  ...
};

其中,也有isa這個成變量, 表示當(dāng)前對象屬于哪個類. 我們看到Block的本質(zhì)是struct __block_impl, 它的實(shí)現(xiàn)中有一個isa指針,它也指向當(dāng)前Block是具體哪個類. 因此,如果用面向?qū)ο蟮乃季S方式, 我們認(rèn)為Block也是一個, 只是C語言是面向過程的語言,要實(shí)現(xiàn)面向?qū)ο蟮姆绞?只能通過這種用結(jié)構(gòu)體方式存儲成員變量,用靜態(tài)函數(shù)表示成員方法的形式來表示.

與此同時(shí),我們看到Block實(shí)際執(zhí)行的函數(shù)__main_block_func_0用的參數(shù)是Block本身 -- __cself, 這與OC/C++中的方法調(diào)用類似, OC/C++中成員函數(shù)的第一個參數(shù)實(shí)際上是self, 表示類本身, 我們前面理解到Block實(shí)際是一個,那么對于它的實(shí)例方法的第一個參數(shù)就是self也不難理解了.

因此, 我們用面向?qū)ο蟮乃季S,可以這樣理解Block:

  1. struct __block_impl是所有Block抽象類,抽象類中規(guī)定了Block中的幾個關(guān)鍵的成員變量isa表示Block的具體的根類型(后面會講到Block有3種根類型),略過Flagsreserved成員變量, 而FuncPtr是一個函數(shù)指針, 可以認(rèn)為是Block類中的成員函數(shù).
  2. struct __main_block_impl_0是實(shí)際項(xiàng)目中我們寫的Block的實(shí)際類,我們創(chuàng)建Block時(shí), 實(shí)際是創(chuàng)建的該類的實(shí)例.
  3. 調(diào)用Block方法時(shí), 實(shí)際調(diào)用的實(shí)際類的實(shí)例的成員函數(shù)FuncPtr.
  4. 用偽碼寫成結(jié)果如下:
Class VirtualBlock{
  Class isa;
  int Flags;
  int reserved;

  void FuncPtr(); // 實(shí)際Block中的函數(shù)指針
}

Class ConcreteBlock: VirtualBlock {
  ConcreteBlockDesc desc;
  ConcretBlock(...);// 構(gòu)造函數(shù)
}

Class ConcreteBlockDesc {
  int reserved;
  int block_size;
  ... // 其他成員函數(shù)
}

因此這里我們可以做一個小結(jié), Block本質(zhì)是一個結(jié)構(gòu)體, 或者本質(zhì)上是一個類似與OC類的結(jié)構(gòu)體類, 因?yàn)樗?code>isa指針,self指針, 成員變量, 和成員函數(shù). 唯一與真正的類的區(qū)別是: 我們創(chuàng)建的結(jié)構(gòu)體存儲在應(yīng)用程序的上或者, 而OC的類內(nèi)存內(nèi)容存儲在堆上.(后面我們會知道Block也能存儲在上)

<<Objective-C 高級編程: iOS與OSX多線程和內(nèi)存管理>>中有講到C++的self,與Objective-C的self的底層實(shí)現(xiàn).

參考資料

  1. <<Objective-C 高級編程: iOS與OSX多線程和內(nèi)存管理>>
  2. https://blog.csdn.net/deft_mkjing/article/details/53149629
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖抬闯,帶你破解...
    沈念sama閱讀 211,348評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡星岗,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,122評論 2 385
  • 文/潘曉璐 我一進(jìn)店門戒洼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來俏橘,“玉大人,你說我怎么就攤上這事圈浇×绕” “怎么了?”我有些...
    開封第一講書人閱讀 156,936評論 0 347
  • 文/不壞的土叔 我叫張陵磷蜀,是天一觀的道長召耘。 經(jīng)常有香客問我,道長褐隆,這世上最難降的妖魔是什么污它? 我笑而不...
    開封第一講書人閱讀 56,427評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮庶弃,結(jié)果婚禮上衫贬,老公的妹妹穿的比我還像新娘。我一直安慰自己歇攻,他們只是感情好祥山,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,467評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著掉伏,像睡著了一般缝呕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上斧散,一...
    開封第一講書人閱讀 49,785評論 1 290
  • 那天供常,我揣著相機(jī)與錄音,去河邊找鬼鸡捐。 笑死栈暇,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的箍镜。 我是一名探鬼主播源祈,決...
    沈念sama閱讀 38,931評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼煎源,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了香缺?” 一聲冷哼從身側(cè)響起手销,我...
    開封第一講書人閱讀 37,696評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎图张,沒想到半個月后锋拖,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,141評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡祸轮,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,483評論 2 327
  • 正文 我和宋清朗相戀三年兽埃,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片适袜。...
    茶點(diǎn)故事閱讀 38,625評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡柄错,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出苦酱,到底是詐尸還是另有隱情售貌,我是刑警寧澤,帶...
    沈念sama閱讀 34,291評論 4 329
  • 正文 年R本政府宣布躏啰,位于F島的核電站,受9級特大地震影響耙册,放射性物質(zhì)發(fā)生泄漏给僵。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,892評論 3 312
  • 文/蒙蒙 一详拙、第九天 我趴在偏房一處隱蔽的房頂上張望帝际。 院中可真熱鬧,春花似錦饶辙、人聲如沸蹲诀。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽脯爪。三九已至,卻和暖如春矿微,著一層夾襖步出監(jiān)牢的瞬間痕慢,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工涌矢, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留掖举,地道東北人。 一個月前我還...
    沈念sama閱讀 46,324評論 2 360
  • 正文 我出身青樓娜庇,卻偏偏與公主長得像塔次,于是被迫代替她去往敵國和親方篮。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,492評論 2 348

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

  • 1 Block機(jī)制 (Very Good) Block技巧與底層解析 http://www.reibang.com...
    Kevin_Junbaozi閱讀 4,038評論 3 48
  • 前言 Blocks是C語言的擴(kuò)充功能励负,而Apple 在OS X Snow Leopard 和 iOS 4中引入了這...
    小人不才閱讀 3,759評論 0 23
  • Block基礎(chǔ)回顧 1.什么是Block藕溅? 帶有局部變量的匿名函數(shù)(名字不重要,知道怎么用就行)熄守,差不多就與C語言...
    Bugfix閱讀 6,753評論 5 61
  • 1.Block的實(shí)現(xiàn) 我們在命令行下輸入clang -rewrite-objc 源代碼文件名就可以把含有block...
    雪山飛狐_91ae閱讀 372評論 0 2
  • 這是一張讓譚心小盆友感覺很驕傲的獎狀蜈垮,拿出來給我們看時(shí),一臉的自豪:爸爸媽媽裕照,這是我們寫作文比賽的獎狀攒发,全班只有一...
    吳麗濱閱讀 220評論 4 2