美文网首页iOS基础·OC高级篇
block 是什么?如何使用?底层是怎样的?

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

作者: 海牛骑士 | 来源:发表于2019-03-26 23:43 被阅读0次

一.初识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捕获进行指针传递

相关文章

网友评论

    本文标题:block 是什么?如何使用?底层是怎样的?

    本文链接:https://www.haomeiwen.com/subject/lfcgvqtx.html