美文网首页
你是我的面试吗?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