探尋block

block的本質(zhì)

block本質(zhì)上也是一個(gè)OC對象闷堡,它內(nèi)部也有個(gè)isa指針喉前。
block是封裝了函數(shù)調(diào)用以及函數(shù)調(diào)用環(huán)境的OC對象佳吞。

//
//  Created by SK on 2020/7/7.
//  Copyright ? 2020 SK. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "Person.h"

NSString *name = @"jack";

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        int age = 20;
        static int sex = 0;
        Person *person1 = [[Person alloc] init];
        Person *person2 = [[Person alloc] init];
        Person *person3 = [[Person alloc] init];
        
        __strong Person *strongPerson = person1;
        __weak Person *weakPerson = person2;
        __block Person *blockPerson = person3;
        
        void (^block)(void) = ^{
            NSLog(@"%d-%d-%@", age, sex, name);
            [strongPerson run];
            [weakPerson run];
            [blockPerson run];
        };
        block();
        
    }
    return 0;
}

我們通過執(zhí)行
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc -fobjc-arc -fobjc-runtime=ios-9.0 main.m
編譯一下上述代碼苍姜,來看一下block的底層結(jié)構(gòu)

struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};

struct __Block_byref_blockPerson_0 {
  void *__isa;
__Block_byref_blockPerson_0 *__forwarding;
 int __flags;
 int __size;
 void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
 Person *__strong blockPerson;
};

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int age;
  int *sex;
  Person *__strong strongPerson;
  Person *__weak weakPerson;
  __Block_byref_blockPerson_0 *blockPerson; // by ref
}


static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  __Block_byref_blockPerson_0 *blockPerson = __cself->blockPerson; // bound by ref
  int age = __cself->age; // bound by copy
  int *sex = __cself->sex; // bound by copy
  Person *__strong strongPerson = __cself->strongPerson; // bound by copy
  Person *__weak weakPerson = __cself->weakPerson; // bound by copy
            NSLog((NSString *)&__NSConstantStringImpl__var_folders_h6_tk2fn5rd5r92tqhsv84hkj540000gn_T_main_e976c4_mi_1, age, (*sex), name);
            ((void (*)(id, SEL))(void *)objc_msgSend)((id)strongPerson, sel_registerName("run"));
            ((void (*)(id, SEL))(void *)objc_msgSend)((id)weakPerson, sel_registerName("run"));
            ((void (*)(id, SEL))(void *)objc_msgSend)((id)(blockPerson->__forwarding->blockPerson), sel_registerName("run"));
}

static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->strongPerson, (void*)src->strongPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);_Block_object_assign((void*)&dst->weakPerson, (void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);_Block_object_assign((void*)&dst->blockPerson, (void*)src->blockPerson, 8/*BLOCK_FIELD_IS_BYREF*/);}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->strongPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);_Block_object_dispose((void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);_Block_object_dispose((void*)src->blockPerson, 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(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 

        int age = 20;
        static int sex = 0;
        Person *person1 = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc")), sel_registerName("init"));
        Person *person2 = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc")), sel_registerName("init"));
        Person *person3 = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc")), sel_registerName("init"));

        __attribute__((objc_ownership(strong))) Person *strongPerson = person1;
        __attribute__((objc_ownership(weak))) Person *weakPerson = person2;
        __attribute__((__blocks__(byref))) __Block_byref_blockPerson_0 blockPerson = {(void*)0,(__Block_byref_blockPerson_0 *)&blockPerson, 33554432, sizeof(__Block_byref_blockPerson_0), __Block_byref_id_object_copy_131, __Block_byref_id_object_dispose_131, person3};

        void (*block)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, age, &sex, strongPerson, weakPerson, (__Block_byref_blockPerson_0 *)&blockPerson, 570425344));
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);

    }
    return 0;
}

從上述代碼可以看出译蒂,block的底層結(jié)構(gòu)如下:


Block底層結(jié)構(gòu).png

block的類型

block有3種類型,可以通過調(diào)用class方法或者isa指針查看具體類型谊却,最終都是繼承自NSBlock類型柔昼。

  • NSGlobalBlock ( _NSConcreteGlobalBlock )
    沒有訪問auto變量
  • NSStackBlock ( _NSConcreteStackBlock )
    訪問了auto變量
  • NSMallocBlock ( _NSConcreteMallocBlock )
    NSStackBlock調(diào)用了copy

block的變量捕獲

為了保證block內(nèi)部能夠正常訪問外部的變量,block有個(gè)變量捕獲機(jī)制


image.png
  • _NSConcreteGlobalBlock類型的block調(diào)動copy炎辨,無變化
  • _NSConcreteStackBlock類型的block調(diào)用copy捕透,從棧復(fù)制到堆
  • _NSConcreteMallocBlock類型的block調(diào)用copy, 引用計(jì)數(shù)增加

在ARC環(huán)境下,編譯器會根據(jù)情況自動將棧上的block復(fù)制到堆上碴萧,例如:

  • block作為函數(shù)返回值時(shí)
  • 將block賦值給__strong指針時(shí)
  • block作為Cocoa API中方法名含有usingBlock的方法參數(shù)時(shí)
  • block作為GCD API的方法參數(shù)時(shí)

Auto類型變量

當(dāng)block內(nèi)部訪問了對象類型的auto變量時(shí)

  • 如果block是在棧上乙嘀,將不會對auto變量產(chǎn)生強(qiáng)引用
  • 如果block被拷貝到堆上,會調(diào)用block內(nèi)部的copy函數(shù)
    copy函數(shù)內(nèi)部會調(diào)用_Block_object_assign函數(shù)
    _Block_object_assign函數(shù)會根據(jù)auto變量的修飾符(__strong破喻、__weak虎谢、__unsafe_unretained)做出相應(yīng)的操作,形成強(qiáng)引用(retain)或者弱引用
  • 如果block從堆上移除曹质,會調(diào)用block內(nèi)部的dispose函數(shù)
    dispose函數(shù)內(nèi)部會調(diào)用_Block_object_dispose函數(shù)
    _Block_object_dispose函數(shù)會自動釋放引用的auto變量(release)

__block修飾符

  • __block可以用于解決block內(nèi)部無法修改auto變量值的問題
  • __block不能修飾全局變量婴噩、靜態(tài)變量(static)
  • 編譯器會將__block變量包裝成一個(gè)對象
struct __Block_byref_blockPerson_0 {
  void *__isa;
__Block_byref_blockPerson_0 *__forwarding;
 int __flags;
 int __size;
 void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
 Person *__strong blockPerson;
};

__block的內(nèi)存管理

  • 當(dāng)block在棧上時(shí),并不會對__block變量產(chǎn)生強(qiáng)引用
  • 當(dāng)block被copy到堆時(shí)
    • 會調(diào)用block內(nèi)部的copy函數(shù)
    • copy函數(shù)內(nèi)部會調(diào)用_Block_object_assign函數(shù)
    • _Block_object_assign函數(shù)會對__block變量形成強(qiáng)引用(retain)
  • 當(dāng)block從堆中移除時(shí)
    • 會調(diào)用block內(nèi)部的dispose函數(shù)
    • dispose函數(shù)內(nèi)部會調(diào)用_Block_object_dispose函數(shù)
    • _Block_object_dispose函數(shù)會自動釋放引用的__block變量(release)

循環(huán)引用

__weak typeof(self) weakSelf = self;
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末羽德,一起剝皮案震驚了整個(gè)濱河市几莽,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌宅静,老刑警劉巖章蚣,帶你破解...
    沈念sama閱讀 212,718評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異姨夹,居然都是意外死亡纤垂,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評論 3 385
  • 文/潘曉璐 我一進(jìn)店門磷账,熙熙樓的掌柜王于貴愁眉苦臉地迎上來峭沦,“玉大人,你說我怎么就攤上這事够颠∥跏蹋” “怎么了榄鉴?”我有些...
    開封第一講書人閱讀 158,207評論 0 348
  • 文/不壞的土叔 我叫張陵履磨,是天一觀的道長。 經(jīng)常有香客問我庆尘,道長剃诅,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,755評論 1 284
  • 正文 為了忘掉前任驶忌,我火速辦了婚禮矛辕,結(jié)果婚禮上笑跛,老公的妹妹穿的比我還像新娘。我一直安慰自己聊品,他們只是感情好飞蹂,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,862評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著翻屈,像睡著了一般陈哑。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上伸眶,一...
    開封第一講書人閱讀 50,050評論 1 291
  • 那天惊窖,我揣著相機(jī)與錄音,去河邊找鬼厘贼。 笑死界酒,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的嘴秸。 我是一名探鬼主播毁欣,決...
    沈念sama閱讀 39,136評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼赁遗!你這毒婦竟也來了署辉?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,882評論 0 268
  • 序言:老撾萬榮一對情侶失蹤岩四,失蹤者是張志新(化名)和其女友劉穎哭尝,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體剖煌,經(jīng)...
    沈念sama閱讀 44,330評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡材鹦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,651評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了耕姊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片桶唐。...
    茶點(diǎn)故事閱讀 38,789評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖茉兰,靈堂內(nèi)的尸體忽然破棺而出尤泽,到底是詐尸還是另有隱情,我是刑警寧澤规脸,帶...
    沈念sama閱讀 34,477評論 4 333
  • 正文 年R本政府宣布坯约,位于F島的核電站,受9級特大地震影響莫鸭,放射性物質(zhì)發(fā)生泄漏闹丐。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,135評論 3 317
  • 文/蒙蒙 一被因、第九天 我趴在偏房一處隱蔽的房頂上張望卿拴。 院中可真熱鬧衫仑,春花似錦、人聲如沸堕花。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,864評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽缘挽。三九已至如贷,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間到踏,已是汗流浹背杠袱。 一陣腳步聲響...
    開封第一講書人閱讀 32,099評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留窝稿,地道東北人楣富。 一個(gè)月前我還...
    沈念sama閱讀 46,598評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像伴榔,于是被迫代替她去往敵國和親纹蝴。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,697評論 2 351