iOS - Block類型簡(jiǎn)介

概述

代碼塊Block是蘋(píng)果在iOS4開(kāi)始引入的對(duì)C語(yǔ)言的擴(kuò)展,用來(lái)實(shí)現(xiàn)匿名函數(shù)的特性,Block是一種特殊的數(shù)據(jù)類型,其可以正常定義變量却特、作為參數(shù)堪藐、作為返回值,特殊地,Block還可以保存一段代碼,在需要的時(shí)候調(diào)用,目前Block已經(jīng)廣泛應(yīng)用于iOS開(kāi)發(fā)中,常用于GCD曾棕、動(dòng)畫(huà)获列、排序及各類回調(diào)

block 會(huì)在編譯過(guò)程中粱胜,會(huì)被當(dāng)做結(jié)構(gòu)體進(jìn)行處理烛占。 其結(jié)構(gòu)Block-ABI-Apple大概是這樣的:

struct Block_literal_1 {
   void *isa; // initialized to &_NSConcreteStackBlock or &_NSConcreteGlobalBlock
   int flags;
   int reserved;
   void (*invoke)(void *, ...);
   struct Block_descriptor_1 {
   unsigned long int reserved;         // NULL
       unsigned long int size;         // sizeof(struct Block_literal_1)
       // optional helper functions
       void (*copy_helper)(void *dst, void *src);     // IFF (1<<25)
       void (*dispose_helper)(void *src);             // IFF (1<<25)
       // required ABI.2010.3.16
       const char *signature;                         // IFF (1<<30)
   } *descriptor;
   // imported variables
};```

isa 指針會(huì)指向 block 所屬的類型,用于幫助運(yùn)行時(shí)系統(tǒng)進(jìn)行處理胁出。
Block 常見(jiàn)的類型有三種离唬,分別是` _NSConcreteStackBlock ` ` _NSConcreteMallocBlock` `_NSConcreteGlobalBlock` 。
另外還包括只在GC環(huán)境下使用的 `_NSConcreteFinalizingBlock` `_NSConcreteAutoBlock`  ` _NSConcreteWeakBlockVariable`划鸽。

下面摘自 [libclosure-65 – Block_private.h-213](http://opensource.apple.com/source/libclosure/libclosure-65/runtime.c)
```objectivec
// the raw data space for runtime classes for blocks
// class+meta used for stack, malloc, and collectable based blocks
BLOCK_EXPORT void * _NSConcreteMallocBlock[32]
   __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_2);
BLOCK_EXPORT void * _NSConcreteAutoBlock[32]
   __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_2);
BLOCK_EXPORT void * _NSConcreteFinalizingBlock[32]
   __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_2);
BLOCK_EXPORT void * _NSConcreteWeakBlockVariable[32]
   __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_2);
// declared in Block.h
// BLOCK_EXPORT void * _NSConcreteGlobalBlock[32];
// BLOCK_EXPORT void * _NSConcreteStackBlock[32];```

##Block的幾種類型
1._NSConcreteGlobalBlock &  _NSConcreteStackBlock
2._NSConcreteMallocBlock
3._NSConcreteFinalizingBlock &  _NSConcreteAutoBlock
4._NSConcreteWeakBlockVariable

### _NSConcreteGlobalBlock &  _NSConcreteStackBlock 
`_NSConcreteGlobalBlock`& ` _NSConcreteStackBlock `是 block 初始化時(shí)設(shè)置的類型。
在以下情況中,block 會(huì)初始化為` _NSConcreteGlobalBlock`:

1.未捕獲外部變量裸诽。
在 [static void computeBlockInfo(CodeGenModule &CGM, CodeGenFunction *CGF,CGBlockInfo &info)](http://clang.llvm.org/doxygen/CGBlocks_8cpp_source.html#l00326) 函數(shù)內(nèi)的 [334行](http://clang.llvm.org/doxygen/CGBlocks_8cpp_source.html#l00334) 至 [339行](http://clang.llvm.org/doxygen/CGBlocks_8cpp_source.html#l00339)嫂用,通過(guò)判斷 block(以及嵌套的block) 是否捕捉了本地存儲(chǔ)(原文為:local storage),未捕獲時(shí)丈冬,block 會(huì)初始化為`_NSConcreteGlobalBlock`
```objectivec
 if (!block->hasCaptures()) {
   info.StructureType =
     llvm::StructType::get(CGM.getLLVMContext(), elementTypes, true);
   info.CanBeGlobal = true;
   return;
 }

2.當(dāng)需要布局(layout)的變量的數(shù)量為0時(shí)嘱函。
static void computeBlockInfo(CodeGenModule &CGM, CodeGenFunction *CGF,CGBlockInfo &info)函數(shù)內(nèi),通過(guò)計(jì)算 block 的布局(layout)埂蕊。當(dāng)需要布局的變量為0時(shí)往弓,block 會(huì)初始化為_NSConcreteGlobalBlock

統(tǒng)計(jì)需要布局(layout)的變量:

  • this (為了訪問(wèn) c++ 的成員變量和函數(shù),需要 this 指針)
  • 依次按下列規(guī)則處理捕獲的變量:
    • 不需要計(jì)算布局的變量:
      • 生命周期為靜態(tài)的變量(被 const static 修飾的變量蓄氧,不被函數(shù)包含的靜態(tài)常量函似,c++中生命周期為靜態(tài)的變量)
      • 函數(shù)參數(shù)
    • 需要計(jì)算布局的變量:被 __block 修飾的變量,以上未提到的類型(比如block)
      Tips:當(dāng)需要布局(layout)的變量的統(tǒng)計(jì)完畢后喉童,會(huì)按照以下順序進(jìn)行一次穩(wěn)定排序撇寞。
  • __strong 修飾的變量
  • ByRef 類型
  • __weak 修飾的變量
  • 其它類型

_NSConcreteMallocBlock

在非垃圾收集環(huán)境下,當(dāng)_NSConcreteStackBlock類型的block 被真正復(fù)制時(shí)堂氯,在_Block_copy_internal 方法內(nèi)部蔑担,會(huì)轉(zhuǎn)換為 _NSConcreteMallocBlock libclosure-65/runtime.c

// Its a stack block.  Make a copy.
if (!isGC) {
   struct Block_layout *result = malloc(aBlock->descriptor->size);
   if (!result) return NULL;
   memmove(result, aBlock, aBlock->descriptor->size); // bitcopy first
   // reset refcount
   result->flags &= ~(BLOCK_REFCOUNT_MASK|BLOCK_DEALLOCATING);    // XXX not needed
   result->flags |= BLOCK_NEEDS_FREE | 2;  // logical refcount 1
   result->isa = _NSConcreteMallocBlock;
   _Block_call_copy_helper(result, aBlock);
   return result;
}```

###_NSConcreteFinalizingBlock & _NSConcreteAutoBlock
在垃圾收集環(huán)境下,當(dāng) block 被復(fù)制時(shí)咽白,如果block 有 ctors & dtors 時(shí)啤握,則會(huì)轉(zhuǎn)換為 `_NSConcreteFinalizingBlock`類型,反之晶框,則會(huì)轉(zhuǎn)換為`_NSConcreteAutoBlock` 類型

if (hasCTOR) {
result->isa = _NSConcreteFinalizingBlock;
}
else {
result->isa = _NSConcreteAutoBlock;
}```

_NSConcreteWeakBlockVariable

GC環(huán)境下排抬,當(dāng)對(duì)象被__weak __block修飾,且從棧復(fù)制到堆時(shí)三妈,block 會(huì)被標(biāo)記為 _NSConcreteWeakBlockVariable
類型畜埋。

bool isWeak = ((flags & (BLOCK_FIELD_IS_BYREF|BLOCK_FIELD_IS_WEAK)) == (BLOCK_FIELD_IS_BYREF|BLOCK_FIELD_IS_WEAK));
// if its weak ask for an object (only matters under GC)
struct Block_byref *copy = (struct Block_byref *)_Block_allocator(src->size, false, isWeak);
copy->flags = src->flags | _Byref_flag_initial_value; // non-GC one for caller, one for stack
copy->forwarding = copy; // patch heap copy to point to itself (skip write-barrier)
src->forwarding = copy;  // patch stack to point to heap copy
copy->size = src->size;
if (isWeak) {
  copy->isa = &_NSConcreteWeakBlockVariable;  // mark isa field so it gets weak scanning
}```
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市畴蒲,隨后出現(xiàn)的幾起案子悠鞍,更是在濱河造成了極大的恐慌,老刑警劉巖模燥,帶你破解...
    沈念sama閱讀 221,198評(píng)論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件咖祭,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡蔫骂,警方通過(guò)查閱死者的電腦和手機(jī)么翰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)辽旋,“玉大人浩嫌,你說(shuō)我怎么就攤上這事檐迟。” “怎么了码耐?”我有些...
    開(kāi)封第一講書(shū)人閱讀 167,643評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵追迟,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我骚腥,道長(zhǎng)敦间,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,495評(píng)論 1 296
  • 正文 為了忘掉前任束铭,我火速辦了婚禮廓块,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘契沫。我一直安慰自己带猴,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布埠褪。 她就那樣靜靜地躺著浓利,像睡著了一般。 火紅的嫁衣襯著肌膚如雪钞速。 梳的紋絲不亂的頭發(fā)上贷掖,一...
    開(kāi)封第一講書(shū)人閱讀 52,156評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音渴语,去河邊找鬼苹威。 笑死,一個(gè)胖子當(dāng)著我的面吹牛驾凶,可吹牛的內(nèi)容都是我干的牙甫。 我是一名探鬼主播,決...
    沈念sama閱讀 40,743評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼调违,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼窟哺!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起技肩,我...
    開(kāi)封第一講書(shū)人閱讀 39,659評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤且轨,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后虚婿,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體旋奢,經(jīng)...
    沈念sama閱讀 46,200評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評(píng)論 3 340
  • 正文 我和宋清朗相戀三年然痊,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了至朗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,424評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡剧浸,死狀恐怖锹引,靈堂內(nèi)的尸體忽然破棺而出矗钟,到底是詐尸還是另有隱情,我是刑警寧澤嫌变,帶...
    沈念sama閱讀 36,107評(píng)論 5 349
  • 正文 年R本政府宣布真仲,位于F島的核電站,受9級(jí)特大地震影響初澎,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜虑凛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評(píng)論 3 333
  • 文/蒙蒙 一碑宴、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧桑谍,春花似錦延柠、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,264評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至雹仿,卻和暖如春增热,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背胧辽。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,390評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工峻仇, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人邑商。 一個(gè)月前我還...
    沈念sama閱讀 48,798評(píng)論 3 376
  • 正文 我出身青樓摄咆,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親人断。 傳聞我的和親對(duì)象是個(gè)殘疾皇子吭从,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評(píng)論 2 359

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