Block底层原理三-类型

Block类型

block有3种类型,可以通过调用class方法或者isa指针查看具体类型,最终都是继承自NSBlock类型,NSBlock继承与NSObject

  • NSGlobalBlock ( _NSConcreteGlobalBlock )
  • NSStackBlock ( _NSConcreteStackBlock )
  • NSMallocBlock ( _NSConcreteMallocBlock )

先上代码 再来分析把

- (void)viewDidLoad {
    [super viewDidLoad];
    
    void (^block)(void) = ^{
       NSLog(@"--------");
    };
    
    int age = 10;
    void (^block1)(void) = ^{
        NSLog(@"%d",age);
    };
    
    block();
    block1();
    
    NSLog(@"block is: %@",block);
    NSLog(@"block is: %@",block1);
    NSLog(@"block is: %@",[^{
        NSLog(@"%d",age);
    } class]);
}

//打印结果

2018-08-16 18:46:12.503194+0800 Test[15733:832689] --------
2018-08-16 18:46:12.503402+0800 Test[15733:832689] 10
2018-08-16 18:46:12.503599+0800 Test[15733:832689] block is: <__NSGlobalBlock__: 0x103555090>
2018-08-16 18:46:12.503798+0800 Test[15733:832689] block is: <__NSMallocBlock__: 0x60000004b4c0>
2018-08-16 18:46:12.504141+0800 Test[15733:832689] block is: __NSStackBlock__

//程序的内存分配


image.png
  • NSGlobalBlock:既没有创建对象也没有局部变量的时候,他是NSGlobalBlock类型,他存在内存的数据区域
  • NSMallocBlock:堆区,动态分配内存,alloc,堆区的特点要手动释放,ARC环境下不用管
  • NSStackBlock: 栈区,放局部变量,栈的特点会自动分配内存,离开作用域会自动销毁
block的父类
- (void)viewDidLoad {
    [super viewDidLoad];
    
    void (^block)(void) = ^{
       NSLog(@"--------");
    };
    
  
    block();
    NSLog(@"%@",[block class]);
    NSLog(@"%@",[[block class] superclass]);
    NSLog(@"%@",[[[block class] superclass] superclass]);
    NSLog(@"%@",[[[[block class] superclass] superclass] superclass]);
}

//打印结果
2018-08-16 18:55:37.953707+0800 Test[15816:846355] --------
2018-08-16 18:55:37.953938+0800 Test[15816:846355] __NSGlobalBlock__
2018-08-16 18:55:37.954237+0800 Test[15816:846355] __NSGlobalBlock
2018-08-16 18:55:37.954798+0800 Test[15816:846355] NSBlock
2018-08-16 18:55:37.955377+0800 Test[15816:846355] NSObject
  • block的父类是NSBlock,NSBlock继承于NSObject
区分block类型
  • 没有访问auto变量 就是NSGlobalBlock类型
  • 访问了auto变量 就是NSStackBlock
  • NSStackBlock 调用了copy 就是NSMallocBlock
//下面代码是在MRC情况下编译的,因为ARC会自动调用copy,后面会讲到
- (void)viewDidLoad {
    [super viewDidLoad];
    
    void (^block)(void) = ^{
       NSLog(@"--------");
    };
    
    int age = 10;
    void (^block1)(void) = ^{
        NSLog(@"%d",age);
    };
    
   
    void (^block2)(void) = [^{
         NSLog(@"%d",age);
    }copy];
    
    block();
    block1();
    block2();
    NSLog(@"block is: %@",[block class]);
    NSLog(@"block is: %@",[block1 class]);
    NSLog(@"block is: %@",[block2 class]);
}

//打印结果
2018-08-16 19:03:10.653042+0800 Test[15918:859044] block is: __NSGlobalBlock__
2018-08-16 19:03:10.653448+0800 Test[15918:859044] block is: __NSStackBlock__
2018-08-16 19:03:10.653609+0800 Test[15918:859044] block is: __NSMallocBlock__

进入c++文件看看里面是什么

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc viewContrller.m
struct __ViewController__viewDidLoad_block_impl_0 {
  struct __block_impl impl;
  struct __ViewController__viewDidLoad_block_desc_0* Desc;
  __ViewController__viewDidLoad_block_impl_0(void *fp, struct __ViewController__viewDidLoad_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

struct __ViewController__viewDidLoad_block_impl_1 {
  struct __block_impl impl;
  struct __ViewController__viewDidLoad_block_desc_1* Desc;
  int age;
  __ViewController__viewDidLoad_block_impl_1(void *fp, struct __ViewController__viewDidLoad_block_desc_1 *desc, int _age, int flags=0) : age(_age) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

struct __ViewController__viewDidLoad_block_impl_2 {
  struct __block_impl impl;
  struct __ViewController__viewDidLoad_block_desc_2* Desc;
  int age;
  __ViewController__viewDidLoad_block_impl_2(void *fp, struct __ViewController__viewDidLoad_block_desc_2 *desc, int _age, int flags=0) : age(_age) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
  • 会发现所有的block的isa指向的类型都是_NSConcreteStackBlock类型
  • 说明clang出来的代码并不是我们最终OC写的代码,这只是编译运行时的结果

接下来继续上代码

WKBlock myblock(){
    
    WKBlock block = ^{
        NSLog(@"------");
    };
    return block;
}

int main(int argc, const char * argv[]) {
    @autoreleasepool {
       
        WKBlock block = myblock();
        block();
        
    }
    return 0;
}
//打印结果
2018-08-16 19:12:40.518275+0800 mhhhh[16069:873326] ------
Program ended with exit code: 0

如果是MRC环境下

  • 就有问题 block是存在于栈区(就会危险)
  • 过了作用域就会释放

ARC环境下会自动copy

  WKBlock block = ^[{
        NSLog(@"------");
    } copy];
    return block;
}

int main(int argc, const char * argv[]) {
    @autoreleasepool {
       
        WKBlock block = myblock();
        block();
        
    }
    return 0;
}
堆和栈的区别
  • 地址上来看 栈是高地址 堆是低地址
  • 栈会自动释放,过了作用域立即释放
  • 堆需要手动释放
  • 栈吃了吐,堆吃了拉
__weak,__strong、__block、__unsafe_unretained的区别
__weak, __strong
#import <Foundation/Foundation.h>
#import "WKPerson.h"

typedef void (^WKBlock)(void);

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

    WKBlock block;
     {
        WKPerson *person = [[WKPerson alloc] init];
        person.age = 10;
         __weak WKPerson *weakPerson = person;
        block = ^{
            NSLog(@"---------%d", weakPerson.age);
    };
    
    
    NSLog(@"------");
}

    return 0;
}


#import <Foundation/Foundation.h>

@interface WKPerson : NSObject

@property (assign, nonatomic) int age;

@end


#import "WKPerson.h"

@implementation WKPerson

- (void)dealloc
 {
   
    NSLog(@"WKPerson - dealloc");
 }

@end


2018-08-17 18:14:43.170714+0800 cestess[14399:659983] ------
2018-08-17 18:14:43.170964+0800 cestess[14399:659983] WKPerson - dealloc
Program ended with exit code: 0

我们打印C++文件看看

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
//报错了 说不能编译__weak
 error: 
      cannot create __weak reference because the current deployment target does
      not support weak references
         __attribute__((objc_ownership(weak))) WKPerson *weakPerson = person;

//解决方案 换成下面的指令 支持ARC、指定运行时系统版本,比如
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc -fobjc-arc -fobjc-runtime=ios-8.0.0 main.m

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  WKPerson *__weak weakPerson;
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, WKPerson *__weak _weakPerson, int flags=0) : weakPerson(_weakPerson) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  WKPerson *__weak weakPerson = __cself->weakPerson; // bound by copy

            NSLog((NSString *)&__NSConstantStringImpl__var_folders_v2_sw9667rd2fn8s8hvf6pqn3ph0000gn_T_main_3620fe_mi_0, ((int (*)(id, SEL))(void *)objc_msgSend)((id)weakPerson, sel_registerName("age")));
    }
static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->weakPerson, (void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->weakPerson, 3/*BLOCK_FIELD_IS_OBJECT*/);}

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};
  • 我来解析一下这段代码

  • 因为是ARC环境下 block会默认copy操作,也就是在堆上

  • WKPerson *__weak weakPerson; 外部传的__weak 这里会生成__weak 对象

  • __main_block_desc_0 当有对象的时候 会多生成两个函数

 void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*); copy指向的是__main_block_copy_0 这个函数
void (*dispose)(struct __main_block_impl_0*);  dispose 指向的是__main_block_dispose_0这个函数
  • 当block copy的时候会调用
void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*);这个函数

void (copy)(struct __main_block_impl_0, struct __main_block_impl_0*);调用的时候内部会执行

内部会调用 static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->weakPerson, (void*)src->weakPerson,  3/*BLOCK_FIELD_IS_OBJECT*/);}

_Block_object_assign 会对person对象进行引用

  • 外面传的是__strong就强引用
  • 外面传的__weak就弱引用

如果刚刚是MRC会先释放后打印,为啥因为block已经释放了

__block

首先来说修改block内部对象的值

#import <Foundation/Foundation.h>

typedef void (^WKBlock)(void);

int main(int argc, const char * argv[]) {
        
        __block int age = 10;
   // static int age = 10; 这样也可以修改 上篇文章讲到了 我就不讲了
        WKBlock  block = ^{
            age = 20;
          NSLog(@"%d",age);
         };
    
         block();


    return 0;
}

2018-08-17 18:46:02.927284+0800 cestess[14723:713812] 20
Program ended with exit code: 0

我们转换为c++看看

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __Block_byref_age_0 *age; // by ref
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_age_0 *_age, int flags=0) : age(_age->__forwarding) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};


struct __Block_byref_age_0 {
  void *__isa;
__Block_byref_age_0 *__forwarding;
 int __flags;
 int __size;
 int age;
};


static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  __Block_byref_age_0 *age = __cself->age; // bound by ref

            (age->__forwarding->age) = 20;
          NSLog((NSString *)&__NSConstantStringImpl__var_folders_v2_sw9667rd2fn8s8hvf6pqn3ph0000gn_T_main_4fe34f_mi_0,(age->__forwarding->age));
         }

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

        __attribute__((__blocks__(byref))) __Block_byref_age_0 age = {(void*)0,(__Block_byref_age_0 *)&age, 0, sizeof(__Block_byref_age_0), 10};

        WKBlock block = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, (__Block_byref_age_0 *)&age, 570425344));

         ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);


    return 0;
}

我来解析一下这段代码

  • 使用了__block 会多一个结构体__Block_byref_age_0
struct __Block_byref_age_0 {
  void *__isa;
__Block_byref_age_0 *__forwarding;
 int __flags;
 int __size;
 int age;
};

未完待续........

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容