前面简单的说了Block可以作为属性,方法参数和方法返回使用。这里更加详细的介绍一下Block,并且结合实际情况对Block进行使用。
-
什么是Block
Block又称作是闭包、块。这两种概念又是什么呢?
Block是一种函数,这种函数在别的函数内部存在,并且可以使用其所在函数内的局部变量。
希望这样的解释稍微容易理解一些。
-
Block的写法和调用
在xcode里面,可以输入inlineBlock获取Block的基本写法:
returnType(^blockName)(parameterTypes) = ^(parameters) {
statements
};
调用的时候,直接使用名字加一对小括号就可以了:
blockName ();
但是这个地方有一个坑,它省略了等号后面的返回类型。一般情况下没有什么问题,但是如果返回的是nil,编译器就会报错。如下图:
我们只要在等号后面加上对应的返回值,即可编译通过。
所以严格意义上,等号后面也应该有返回类型。
-
Block在实际开发中的使用
我们先定义一个类,叫Person。
1. block作为属性
给我们之前定义的person类增加一个属性:
//对Block的修饰:MRC用copy,ARC用Strong
@property (nonatomic, strong) void(^eat)(void);
然后测试一下:
Person *p = [Person new];
//Block作为属性
p.eat = ^{
NSLog(@"吃饭了!");
};
_p = p;
调用:
_p.eat();
控制台输出:
吃饭了
2. block作为方法参数
还是给之前的Person类增加一个新的方法:
.h
//Block作为参数时,blockname不需要写在^后面,直接写在括号后面
- (void)eatWith:(void(^)(void))block;
.m
- (void)eatWith:(void (^)(void))block {
block();
}
测试代码:
//Block作为方法参数
[_p eatWith:^{
NSLog(@"吃了一顿大餐!");
}];
控制台输出:
吃了一顿大餐
3. block作为方法的返回值
.h
//Block作为返回值
- (void(^)(int i))run;
.m
- (void(^)(int i))run {
return ^(int i){
NSLog(@"我走了%d米",i);
};
}
测试调用:
void(^block)(int i)= _p.run;
block(1);
//上面这两行代码可以合并为下面这一行
_p.run(1);//有没有发现这个调用和block作为属性时是一样的,下面会继续分析
控制台输出:
我走了1米
-
关于Block用法的引申思考
在前面,我们简单的运用了block的几种使用场景,但是在使用过程中,逐渐体会到了block的神奇。
1. 函数式编程
把Block当做函数的参数,可以把我们的逻辑和函数放在调用时候的block里面,而不是方法内部。这样会让我们在写代码的时候,把相关的逻辑都放在一起,提高了开发效率和程序的可读性。这其实就是函数式编程思想。函数式编程在很多第三方框架中都有明显的体现,比如说我们频繁使用的AFNetWorking、Masonry等。
我们这里对Masonry框架进行一些简单的分析。
Masonry是一个轻量级的布局框架。这是一个非常简单并且常见的Masonry框架的运用:
[view1 mas_makeConstraints:^(MASConstraintMaker *make) {
make.edges.equalTo(superview).with.insets(padding);
}];
2.链式调用
外面的block就是一个函数式编程的体现,但是我们看到它在Block内部调用方法的时候,不是像我们使用常规的OC语言一样,而是不断地使用点语法调用,极大的精简了代码量,同时也拥有很好的可读性,使用起来极其方便。这种方式叫链式调用。
那么结合我们刚刚对block的使用,链式调用时怎么实现的呢?
- 后面带括号,说明方法的返回值是一个Block。
- 调用方法肯定是对象才可以进行调用,说明Block的返回值是一个对象。
- 点语法则说明这个方法没有参数。
结合以上三点思考,我们可以得出一个结论,一个没有参数&有返回值&返回值是Block&Block的返回值是方法的调用者的方法,就可以实现链式调用:
.h
//链式调用方法
- (Person *(^)(NSString *city))travel;
.m
- (Person *(^)(NSString *))travel {
return ^(NSString *city){
NSLog(@"我去了%@",city);
return self;
};
}
测试一下:
//链式调用
_p.travel(@"重庆").travel(@"北京");
控制台输出:
我去了重庆
我去了北京
这样我们就实现了链式调用。
函数式编程和链式调用就是Masonry用到的主要思想。我用这两种思想,写了一个简单的计算器。可以通过链式调用的方式做运算,代码和之前的demo都放在git上点击前往。