美文网首页
你是我的面试吗?iOS开发之Block

你是我的面试吗?iOS开发之Block

作者: 绝不知火 | 来源:发表于2018-04-02 16:40 被阅读0次

    什么是Block

    block可以理解为匿名函数(匿名函数:不以文件形式驻留在文件上,生成方式最简洁,可以在指令窗或任何函数体内通过指令直接生成.)

    block也可以是一种特殊的数据类型,block可以正常定义变量,作为参数,作为返回值,可以保存一段代码,在需要的时候调用.

    block常用于GCD,动画,排序以及各类回调

    Block表达式语法

    ^返回值类型(参数列表){ 表达式 };
    例:

    ^int(int count) {
        return count + 1;
     };
    

    声明Block类型变量语法

    返回值类型(^变量名)(参数列表) = Block表达式
    例:

    int(^ablock)(int) = ^(int year) {
        return year + 1;
    };
    

    当block类型变量作为函数参数时

    - (void)func:(int(^)(int))blk {
        NSLog(@"I love you %@ years", blk);
    }
    // 借助typedef简化block
    typedef int(^blk)(int);
    - (void)func:(blk)blk {
        NSLog(@"I love you %@ years", blk);
    }
    // 1. 声明一个block变量
    int (^addBlock)(int) = ^(int x) {
        return x + 1;
    };
    // 2. 以block作为函数参数,把block像对象一样传递
    [self func:addBlock];
    // 3. 1与2合并一起,以内联定义的Block作为函数参数
    [self func:^(int x) {
        return x + 1;
    }];
    

    block类型变量作返回值时

    - (int(^)(int))funcReturn {
        return ^(int year) {
            return year + 1;
        };
    }
    // 借助typedef简化block
    typedef int(^blk)(int);
    - (blk)funcReturn {
        return ^(int year) {
            return year + 1;
        };
    }
    

    Blockb内存

    声明Block属性的时候为什么用copy

    Block有3种类型
    全局块(NSGlobalBlock)
    栈块(NSStackBlock)
    堆块(NSMallocBlock)

    全局块存储在静态区(也叫全局区),
    栈块存储在栈区,超出作用域则马上被销毁
    堆块存储在堆区中,是一个引用计数的对象,需要自行管理其内存

    怎么判断block的所在位置呢?
    1. block不访问外界变量
      block既不在栈中也不在堆中,此时就为全局块,ARC和MRC下都是如此.
    2. block访问外界变量
      MRC环境下: 访问外界变量的block默认存储在栈区.
      ARC环境下: 访问外界变量的block默认存放在堆中,实际上是先放在栈区,在ARC情况下自动又拷贝到堆区,自动释放.

    使用copy修饰符的作用就是将block从栈区拷贝到堆区,从而延长block的生命周期

    例子1:

    int testInt = 10;
    void (^testBlock)(void) = ^{
        NSLog(@"testInt = %d", testInt);
    };
    testInt = 20;
    testBlock();
    

    例子2:

    __block int testInt = 10;
    void (^testBlock)(void) = ^{
        NSLog(@"testInt = %d", testInt);
    };
    testInt = 20;
    testBlock();
    

    在例子1中,block会把testInt变量复制为自己私有变量,也就是说block会补货栈上的变量(或指针),将其复制为自己私有的const变量.
    在例子2中,testInt是一个局部变量,存储在栈区的.给testInt加入__block修饰符所起到的作用就是只要观察到该变量被block所持有,就将该变量在栈中的内存地址存放到堆中,此时不管block外部还是内部testInt的内存地址都是一样的.

    Block的循环引用解决方案

    - (void)aBlock {
        __weak typeof (self) weakself = self;
        self.block = ^{
            [weakself doSomething];
        };
    }
    // 但是在并发执行的时候,block的执行是可以抢占的,而且对weakSelf指针的调用时序不同可以导致不同的结果,比如在一个特定的时序下weakSelf可能会变成nil,这个时候在执行doAnotherThing就会造成程序的崩溃。为了避免出现这样的问题,采用__strong的方式来进行避免,更改后:
    - (void)aBlock {
        __weak typeof (self) weakself = self;
        self.block = ^{
            __strong typeof(weakSelf) strongSelf = weakSelf;
           [strongSelf doSomething]; // strongSelf != nil
            // 在抢占的时候,strongSelf还是非nil的。
            [strongSelf doAnotherThing];
        };
    }
    

    Demo

    block回调Demo github

    相关文章

      网友评论

          本文标题:你是我的面试吗?iOS开发之Block

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