OC-Block的高级用法

前面简单的说了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上点击前往。

你可能感兴趣的:(OC-Block的高级用法)