1.什么是Block
带有局部变量的匿名函数,与C语言中的函数指针类似,可当做参数进行传值,且可以没有名字。
格式如下:
- block的代码是内联的,效率高于函数调用;
- block对于外部变量默认是只读属性;
- block被Objective-C看成是对象处理。
2.Block的使用
1.无参数无返回值
//1,无参数,无返回值,声明和定义
void(^MyBlockOne)(void) = ^(void){
NSLog(@"无参数,无返回值");
};
MyBlockOne();//block的调用
2.有参数无返回值
//2,有参数,无返回值,声明和定义
void(^MyblockTwo)(int a) = ^(int a){
NSLog(@"@ = %d我就是block,有参数,无返回值",a);
};
MyblockTwo(100);
3.有参数有返回值
//3,有参数,有返回值
int(^MyBlockThree)(int,int) = ^(inta,intb){
NSLog(@"%d我就是block,有参数,有返回值",a + b);returna + b;
};
MyBlockThree(12,56);
4.无参数有返回值(很少用到)
//4,无参数,有返回值
int(^MyblockFour)(void) = ^{NSLog(@"无参数,有返回值");
return45;
};
MyblockFour();
以上情况都是局部变量的block 参考链接
5.为避免使用同类型block时都要编辑大量代码,可以使用typedef 定义
typedef int (^MyBlock)(int , int);
这时,MyBlock
就成为了一种Block类型
在定义类的属性时可以这样:
@property (nonatomic,copy) MyBlock myBlockOne;
使用时:
self.myBlockOne = ^int (int ,int){
//TODO
}
现在假设一个使用场景,B页面要反向传递一个值到A页面。在B.h类中定义了一个属性的@property (nonatomic,copy) void (^myBlock)(NSString * data);
,在B.m中调用的时候一般要进行一个判断(学习的时候大家应该都会,啰嗦下~)
if (self. myBlock) {
self. myBlock(//需要传递的对象值);
}
在A.m中接受传递值的时候:
B.myBlock = ^(NSString * data){
//TODO
//data 就是B页面传递到A页面的数据
}
//B 为B页面实例化的对象
6.截获自动变量(自动变量=局部变量)
看下面代码:
int main()
{
int dmy = 256;
int val = 10;
const char *fmt = "val = %d\n";
void (^blk)(void) = ^{printf(fmt, val);};
val = 2;
fmt = "These values were changed. val = %d\n";
blk();
return 0;
}
执行结果:val = 10
解释:在该源代码中,Block语法的表达式使用的是它之前声明的自动变量fmt 和val.Block语法中,Block表达式截获的自动变量,已保存该自动变量瞬间的值。因为Block表达式保存了自动变量的值,所以在执行Block语法之后,即使概念Block中的自动变量的值也不会影响Block执行时自动变量的值。这就是所谓的截获
7.用_ _block修饰符修饰,改变局部变量的值
int val = 0;
void (^block)(void) = ^{
val = 1;
}
block();
NSLog(@"val = %d",val);
它会直接报错 error: variable is not assignable (missing __block type specifier)
这时候就是在告诉您要想改变Block中局部变量,可使用__block 修饰符修饰变量。
__block int val = 0;
void (^block)(void) = ^{
val = 1;
}
block();
NSLog(@"val = %d",val); //val的值为1
还有一种纯属使用的情况,不改变值,如:
id array = [[NSMutableArray alloc] init];
void (^blk)(void) = ^{
id obj = [[NSObject alloc] init];
[array addObject:obj];
};
没有向array赋值,仅仅是使用,是截获到了NSMutableArray类对象的结构体指针,不会报错;当向它赋值时就会编译报错。
8. 使用__weak关键字,避免循环使用造成内存泄漏
假设使用场景:在二级页面B中定义了Block属性@property (nonatomic,copy) void (^stateBlock)(NSString * data);
在B.m中调用了block,self.stateBlock(//需要传递的值);
在A.m中
- (void)buttonAction {
B *myVC = [[B alloc] init];
[self presentViewController:myVC animated:YES completion:^{
}];
__weak typeof(self) weakSelf = self;//防止循环引用
//用属性定义的注意:这里属性是不会自动补全的,方法就会自动补全
[myVC.stateBlock = ^(NSString * data){
weakSelf.labelA.text = data;
}];
}
9. Block的回调
开发者在block没发布前,实现回调基本都是通过代理的方式进行的。比如负责网络请求的原生类NSURLConnection类,通过多个协议方法实现请求中的事件处理。而在最新的环境下,使用的NSURLSession已经采用block的方式处理任务请求了。各种第三方网络请求框架也都在使用block进行回调处理。这种转变很大一部分原因在于block使用简单,逻辑清晰,灵活等原因。接下来我会完成一次网络请求,然后通过block进行回调处理。这些回调包括请求完成
按照returnValue(^blockName)(parameters)的方式进行block的声明未免麻烦了些,我们可以通过关键字typedef来为block起类型名称,然后直接通过类型名进行block的创建:
//DownloadManager.h
#import <Foundation/Foundation.h>
@interface DownloadManager : NSObject <NSURLSessionDownloadDelegate>
// block 重命名
typedef void (^DownloadHandler)(NSData * receiveData, NSError * error);
- (void)downloadWithURL:(NSString *)URL parameters:(NSDictionary *)parameters handler:(DownloadHandler)handler ;
@end
//DownloadManager.m
#import "DownloadManager.h"
@implementation DownloadManager
- (void)downloadWithURL:(NSString *)URL parameters:(NSDictionary *)parameters handler:(DownloadHandler)handler
{
NSURLRequest * request = [NSURLRequest requestWithURL:[NSURL URLWithString:URL]];
NSURLSession * session = [NSURLSession sharedSession];
//执行请求任务
NSURLSessionDataTask * task = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
if (handler) {
dispatch_async(dispatch_get_main_queue(), ^{
handler(data,error);
});
}
}];
[task resume];
}
上面通过封装NSURLSession的请求,传入一个处理请求结果的block对象,就会自动将请求任务放到工作线程中执行实现,我们在网络请求逻辑的代码中调用如下:
- (IBAction)buttonClicked:(id)sender {
#define SOGOUURL @"http://dlsw.baidu.com/sw-search-sp/soft/9d/25765/sogou_mac_32c_V3.2.0.1437101586.dmg"
//下载类
DownloadManager * downloadManager = [[DownloadManager alloc] init];
[downloadManager downloadWithURL: SOGOUURL parameters:nil handler:^(NSData *receiveData, NSError *error) {
if (error) {
NSLog(@"下载失败:%@",error);
}else {
NSLog(@"下载成功,%@",receiveData);
}
}];
}