什么是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的所在位置呢?
- block不访问外界变量
block既不在栈中也不在堆中,此时就为全局块,ARC和MRC下都是如此. - 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];
};
}
网友评论