——Objective-C高级编程读书笔记 p79-p137
Blocks定义
带有自动变量(局部变量)的匿名函数。在计算机科学中,此概念也称作闭包(Closure)、lambda计算等。
Blocks语法
Block语法的BN范式 : ^ 返回值类型 参数列表 表达式
^int (int count){return count +1;}
1.省略返回值类型时,如果表达式中有return语句就使用该返回值类型,如果表达式中没有return语句就使用void类型。
2.如果不使用参数,参数列表也可以省略。
最简式:^ 表达式
^{printf("Blocks\n");}
Block类型变量
在定义C语言函数时,可以将定义函数的地址赋值给函数指针的类型变量中。同样地,在Block语法下,可将Block语法赋值给声明为Block类型的变量中。即源代码中一旦使用了Block语法就相当于生成了可赋值给Block类型变量的“值”。
声明Block类型变量示例:
int (^blk) (int);
自动变量值截获只能保存执行Block语法瞬间的值。保存后就不能改写该值。使用__block说明符的自动变量可以在Block中赋值,该变量被称为__block变量。
定义Block
这里表示定义一个people类型的Block,无返回值,接收NSMutableArray类型的数组以及布尔型的isAlive。
#typedef void (^people) (NSMutableArray *array,Bool isAlive);
即使不使用以上语句,也可以直接定义为成员变量:
@property (nonatomic, strong) void (^people) (NSMutableArray *array,Bool isAlive);
注意,在MRC下,strong应该改为copy或retain。
Block本身是像对象一样可以retain,和release。但是,block在创建的时候,它的内存是分配在栈(stack)上,而不是在堆(heap)上。他本身的作于域是属于创建时候的作用域,一旦在创建时候的作用域外面调用block将导致程序崩溃。
block要用copy修饰,还是用strong
Block的实现
Block的实质为Objecitve-C的对象。
Block循环引用
来看下面这个代码片段,blk为Block类型成员变量。
- (id)init
{
self = [super init];
blk = ^{NSLog(@"self = %@",self);};
return self;
}
在这段代码中,本类对象的成员变量blk持有赋值为Block的强引用,即本类对象持有Block。而init方法中,执行的Block语法使用附有__strong类修饰符的id类型变量self,并且Block语法赋值给了blk,Block语法生成的Block又栈赋值到堆,并持有所使用的self。self持有Block,Block持有self,这就造成了循环引用。
解决方案一:
声明附有__weak修饰符的变量,并将self赋值使用。
- (id)init
{
self = [super init];
id __weak tmp = self;
blk = ^{NSLog(@"self = %@",tmp);};
return self;
}
值得注意的是,在面向iOS4或以下的版本,这里可以用__unsafe_unreained修饰符代替__weak。但是他们是有区别的。
对于__weak,当释放指针指向的对象时,该对象的指针将转换为nil,这是比较安全的行为。而__unsafe_unretain,正如其名称隐藏的含义,尽管释放指针指向的对象时,该指针将继续指向原来的内存。这将会导致应用crash,所以是unsafe。
iOS 高级内存管理:比较__unsafe_unretain、__strong、__weak、__autoreleasing
解决方法二:
__weak typeof(self) weakSelf = self;
Block使用
AController中:
BController *bVc = [BController new];
bVc.block = ^(NSMutableDictionary *dict) {
NSLog(@"Block %@",dict);
};
[self.navigationController pushViewController:allStarVc animated:YES];
BController中:
//BController.h
#import <UIKit/UIKit.h>
typedef void(^Block)(NSMutableDictionary *dict);
@interface BController : UIViewController
@property (nonatomic,copy)Block block;
@end
//BController.m
- (void)pop
{
if (_allstarBlock) {
allstarBlock(self.dict);
}
[self.navigationController popViewControllerAnimated:YES];
}