内存管理篇之Blocks
Blocks 是 C 语言的扩充功能:带有自动变量(局部变量)的匿名函数
1. Block 类型变量
声明如下:
int (^block)(int);
block 实现赋值
int(^block)(int) = ^(int num) {
NSLog(@"%d",num);
return num + 10;
};
block(222);
可以按照如下格式定义一个类型:typedef returnType(^blockName)(parameterType) ;
2. Block 作为函数参数
格式1:- type funckName:(returnType(^)(parameterType)) blockName{}
//定义类型
typedef void(^myblock)(int) ;
//声明属性--可以直接把 block 名当做一个类型使用
@property (nonatomic, copy) myblock block;
//block 作为函数参数方式一
-(void)blockTest:(void(^) (int)) func{
func(2);
}
//block 作为函数参数方式二
-(void)blockTest2:(myblock ) func{
func(2);
}
3. Block 截获局部变量值
block 截获的是 自动变量的瞬间值.也就是说在 block 定义之后改变了 自动变量值,也不会影响 block 执行时候的值了
NSString* str = @"test";
void (^pBlock)(void) = ^{
NSLog(@"str = %@",str);
};
str = @"test2";
pBlock();
打印结果: str = test
4. __block说明符
block 因为截获的只是 变量的瞬间值,因此是不能修改的.如果尝试修改会报编译错误. 如果想在 block 中修改外部变量的值,需要在外部使用__ block 说明符.
这个会报编译错误:
NSString* str = @"test";
void (^pBlock)(void) = ^{
NSLog(@"str = %@",str);
str = @"change test";
};
下面的正常:
__block NSString* str = @"test";
void (^pBlock)(void) = ^{
NSLog(@"str = %@",str);
str = @"change test";
};
5. 截获自动变量
如果 向 block 中截获的自动变量赋值,就会产生编译错误,基本数据类型和 OC 类型都一样,但是调用 OC 对象的方法不会产生编译错误
下面的代码运行正常:
NSMutableArray *arr = [NSMutableArray array];
void (^pBlock)(void) = ^{
[arr addObject:@"test"];
};
pBlock();
下面的代码尝试给 arr赋值,编译异常:
NSMutableArray *arr = [NSMutableArray array];
// __block NSMutableArray *arr = [NSMutableArray array]; //这样会没有异常
void (^pBlock)(void) = ^{
arr = [NSMutableArray array];
};
pBlock();
6. 允许 block 改写截获的自动变量的几种情况
- __block说明符修饰
- 静态变量
- 静态全局变量
- 全局变量
如下面几个例子
int globleParam = 1; //1>全局变量
static int sGlobleParam = 2; //2>静态全局变量
int main(int argc, char * argv[]) {
static int index = 3; //3>静态局部变量
// __block int index = 4; //4>__block 说明符
void (^pBlock)(void) = ^{
globleParam *= 2;
sGlobleParam *= 2;
index = 2;
};
pBlock();