block的定义
// 无参数无返回值
void (^block)(void);
// 无参有返回值
int (^block1)(void);
// 有参有返回
int (^block2)(int number);
// 对block tydpef
typedef void(^valueBlock)(NSString *string);
@property (nonatomic, copy) valueBlock valueBlock;
block内部使用变量
void test1() {
int a = 10;
void (^block)(void) = ^{
NSLog(@"a is %d", a);// a是局部变量,block定义之后a就被销毁了。
};
a = 20;
block();
}
void test2() {
__block int a = 10;// __block修饰会编译成一个struct类型,在arc下会强引用,mrc下是将不会retain,可以避免循环引用。
void (^block)(void) = ^{
NSLog(@"a is %d", a);
};
a = 20;
block();
}
void test3() {
static int a = 10; // static 函数每次被调用,普通局部变量都是重新分配,而静态局部变量保持上次调用的值不变。
void (^block)(void) = ^{
NSLog(@"a is %d", a);
};
a = 20;
block();
}
int a = 10;
void test4() {
void (^block)(void) = ^{
NSLog(@"a is %d", a); // a是全局变量
};
a = 20;
block();
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
test1(); // 10
test2(); // 20
test3(); // 20
test4(); // 20
// 只有普通局部变量是传值,其他情况都是传址。
}
- 如果block访问的外部变量是局部变量,那么就是值传递,外界变了不会影响里面。
- 如果block访问的外部变量是__block或者static修饰或者是全局变量,那么就是指针传递
block的内存管理
- 无论当前环境是ARC还是MRC,只要block没有访问外部变量,block始终在全局区。
- MRC情况下
- block如果访问外部变量,block在栈里
- 不能对block使用retain,否则不能保存在堆里
- 只能使用copy,才能放到堆里。
- ARC情况下
- block如果访问外部变量,block在堆里
- block可以使用copy和strong,并且block是一个对象
block的循环引用
这个代码中shop指向Shop对象,shop属性myBlock指向Block,block代码指向Shop对象出现循环引用,一般Xcode会有提示,这个很好。
参考这篇文章,对于block的循环引用,内存管理,作者写的很详细,我就不写了。
block中的weakself什么时候加
不是什么时候都要加的,只有出现循环引用的时候才需要加,self->block->self.property/self->_ivar这样的循环链时要加weakself
block传值
- 在控制器间传值可以使用代理或者block,使用block相对来说比较简洁。
SecondViewController *secondVC = [[SecondViewController alloc] init];
secondVC.valueBlock = ^(NSString *string) {
NSLog(@"ViewController拿到了SecondVC的值%@", string);
self.textFild.text = string;
};
[self presentViewController:secondVC animated:YES completion:nil];
在secondVC中声明一个block属性
typedef void(^valueBlock)(NSString *string);
@interface SecondViewController : UIViewController
@property (nonatomic, copy) valueBlock valueBlock;
@end
在.m中实现方法
self.valueBlock(@"hello xiaofang");
这样在textfild上就可以显示第二个页面传过来的字符串了。
block作为一个参数使用
在一个类中声明一个带block参数的方法
- (void)calculator:(int(^)(int result))block;
在m文件中实现方法
- (void)calculator:(int(^)(int result))block {
self.result = block(self.result);
NSLog(@"result = %d", self.result);
}
在其他类中调用方法
Shop *shop = [[Shop alloc] init];
// 调用 block用作参数
[shop calculator:^int(int result) {
result += 5;
return result;
}];
block作为返回值使用
在一个类中声明一个返回值是block,并且block的返回值是改类
- (Shop *(^)(int a))add;
m文件中实现方法
- (Shop *(^)(int a))add {
return ^(int a) {
_result += a;
return self;
在其他类中调用
#pragma MARK - block作为返回值使用
- (void)test9 {
Shop *shop = [[Shop alloc] init];
shop.add(1).add(2).add(3);
NSLog(@"%d", shop.result);
}
总结block需要注意的点
- 在block内部使用外部指针且造成循环引用的情况下需要使用
__weak
修饰外部指针typeof(type) weaktype = type;
- 在block内部如果调用了延时函数还使用弱指针会取不到该指针,因为已经被销毁,需要block内部再将弱指针强引用一下
__strong typeof(self) strongSelf = weakSelf;
Shop *shop = [[Shop alloc] init];
shop.string = @"welcom to my shop";
WeakSelf(shop);
shop.myBlock = ^{
StrongSelf(shop);
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSLog(@"%@",shop.string);
});
// 注释掉StrongSelf(shop) 延迟2秒执行发现会娶不到弱指针,打印nill。需要在block内部将弱指针在强引用一下
};
shop.myBlock();
- 如果需要在block内部需要外部变量的话需要使用
__block
修饰外部变量。