block 是什么?如何使用?底层是怎样的?

一.初识Block

block 的本质:oc对象 底层是一段c语言函数(struct)。
block的特性:自动捕获变量。
既然是oc对象 我们就可以通过控制台打印来输出我们的block.

    void(^block)(void) = ^{
    };
    block();
    NSLog(@"我是-----%@",block);
    
    int a = 10;
    void(^block1)(void) = ^{
        NSLog(@"%d",a);
    };
    block1();
    NSLog(@"我是-----%@",block1);
    
    int b =10;
    NSLog(@"我是-----%@",^{
        NSLog(@"%d",b);
        
    });
//输出结果
2019-03-26 13:44:54.902696+0800 hookDemo[24004:6173365] 我是-----<__NSGlobalBlock__: 0x105f8b0c0> //0x1  
2019-03-26 13:44:54.902945+0800 hookDemo[24004:6173365] 10
2019-03-26 13:44:54.903100+0800 hookDemo[24004:6173365] 我是-----<__NSMallocBlock__: 0x600002c4c2a0>
2019-03-26 13:44:54.903215+0800 hookDemo[24004:6173365] 我是-----<__NSStackBlock__: 0x7ffee9c739a0>

根据上面的输出结果我们可以看到输出了三种不同的block它们分别是:

NSglobalBlock : 静态全局block
NSMallocBlock : 堆Block(调用了外部变量之后)
NSStackBlock : 栈Block

二.如何解决block 循环引用 下面代码为例

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    self.tblock = ^{
      NSLog(@"%@",self.userName);
    };
 self.tblock();
}


-(void)dealloc
{
    NSLog(@"我释放了");
    
}
上面这段代码 在该控制器pop的时候 没有执行 dealloc  方法 这是由于我们当前的block和self互相进行了强引用 造成了循环引用导致对象不能释放。
方式一 weak修饰self
   - (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    
    __weak typeof(self) weakSelf = self;
     self.tblock = ^{
        
      NSLog(@"%@",weakSelf.userName);
    };
 self.tblock();
}


-(void)dealloc
{
    NSLog(@"我释放了");
    
}
2019-03-26 15:49:32.304656+0800 hookDemo[25909:6245650] 我释放了

😊通过上面的weak修饰 能解决我们的循环引用 但是当我们self提前释放了之后 看看我们block内部执行会发生什么情况。

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    
    __weak typeof(self) weakSelf = self;
     self.tblock = ^{
        //延迟两秒执行
         dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
             NSLog(@"%@",weakSelf.userName);
         });
    };
 self.tblock();
}
当我们push 页面之后快速pop 发现控制台打印 如下数据
2019-03-26 15:58:14.439173+0800 hookDemo[26055:6251474] 我释放了
2019-03-26 15:58:15.027246+0800 hookDemo[26055:6251474] (null)
😭😭😭😭😭what?  self提前释放了 导致我们block里面无法使用 。
单用weak 修饰 没有完美解决循环引用的问题

改进方式一 在block 内部用strong再对对weak修饰过的self进行修饰
- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    
    __weak typeof(self) weakSelf = self;
     self.tblock = ^{
         
         __strong typeof(weakSelf) strongSelf = weakSelf;
         
        //延迟两秒执行
         dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
             NSLog(@"%@",strongSelf.userName);
         });
         
     
    };
    self.tblock();
    
}

控制台输出:
2019-03-26 16:10:37.165801+0800 hookDemo[26224:6257894] zhangshan
2019-03-26 16:10:37.166009+0800 hookDemo[26224:6257894] 我释放了

😄👌😄 这样就解决了循环引用的问题


其它解决 一 __block 修饰 (切记 修饰完之后再block执行完之后需要把修饰后的对象设置成nil)
- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    
//  __weak typeof(self) weakSelf = self;
    
      __block  testBlockController * bk = self;
     self.tblock = ^(testBlockController * uw){
         
//         __strong typeof(weakSelf) strongSelf = weakSelf;
         
        //延迟两秒执行
         dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
             NSLog(@"%@",bk.userName);
         });
         
     
    };
    self.tblock(self);
    
}


-(void)dealloc
{
    NSLog(@"我释放了");
    
}
打印结果:
2019-03-26 16:22:09.853963+0800 hookDemo[26415:6265512] zhangshan
2019-03-26 16:22:09.854201+0800 hookDemo[26415:6265512] 我释放了
😄😄😄 这样也能打破循环引用。
其它解决方式二 改变思维侧翼出击 参数传递。将我们的self 当初参数进行传递

#import "testBlockController.h"
typedef void(^textBlock)(testBlockController *uw);

@interface testBlockController ()

@property(copy,nonatomic) textBlock tblock;

@property(copy,nonatomic)NSString * userName;

@end

@implementation testBlockController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    // Do any additional setup after loading the view.
    
    self.userName = @"zhangshan";
    
//  __weak typeof(self) weakSelf = self;
    
//      __block  testBlockController * bk = self;
     self.tblock = ^(testBlockController * uw){
         
//         __strong typeof(weakSelf) strongSelf = weakSelf;
         
        //延迟两秒执行
         dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
             NSLog(@"%@",uw.userName);
         });
         
     
    };
    self.tblock(self);
    
}


-(void)dealloc
{
    NSLog(@"我释放了");
    
}
2019-03-26 16:26:37.811508+0800 hookDemo[26484:6268048] zhangshan
2019-03-26 16:26:37.811747+0800 hookDemo[26484:6268048] 我释放了

😄😄😄 这样也能打破循环引用。

三.block 底层分析

新建一个文件夹 mkdir  blockTest      vim block.c   
copy 下面这段代码 一个简单的main函数 和一个block调用。

#include "stdio.h"
int main()
{
 
    void(^block)(void) =^{
        printf("hello world");
    };
    block();
    return 0;
}   
结束 :wq 完成

通过 clang -rewrite-objc block.c -o block.cpp 命令 将我们的.c 转换成我们的.cpp 这样我们就能去查看分析block的底层实现
打开我们的。.cpp 找到我们的main函数

int main()
{

    void(*block)(void) =((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
    ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
    return 0;
}
__main_block_impl_0 这就是我们的block对象    __main_block_func_0 和 __main_block_desc_0_DATA就是传递的参数。

//block 结构体对象 
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;
  }
};

// 没有修饰的变量会被block捕获 进行值传递
//__block 修饰的变量会被block捕获进行指针传递

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

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,076评论 1 32
  • 1.设计模式是什么? 你知道哪些设计模式,并简要叙述?设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类型...
    龍飝阅读 2,127评论 0 12
  • iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实现匿名函数的特性,B...
    smile刺客阅读 2,327评论 2 26
  • block.png iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实...
    全栈农民工阅读 588评论 0 1
  • 这周的践行非常不好,几乎归零。没有日历,清单,没有三只青蛙和A4纸工作法。 1.健康:这周唯一坚持的是早睡早起,没...
    神仙洋洋阅读 183评论 0 0