Objective-C Block,从0到1(1)

作者: 旅行的光 | 来源:发表于2016-07-30 13:56 被阅读238次

“block”? “块”?

相信有C语言编程基础的童鞋们对于block一定不会感到陌生,用咱们汉语说,block就是“块”。对于从没接触过block的童鞋,大家一定会很疑惑到底什么是“块”,难道就是一坨代码吗?从某种角度来说“一坨”这个词用的并没有错。那么让我们看看什么是“块”吧!

“块” 是一种可在C,C++及Objective-C代码中使用的“词法闭包”,开发者可将代码像对象一样传递,令其在不同环境下运行。在定义“块”的范围内,它可以访问到其中的全部变量。

如何定义block?

1,最简单的block

^{
  //我就是个block
};

看到这样的花括号是不是觉得很奇葩,没错这就是一个没有变量名,没有返回值,没有参数的“三无block”。但是大家一定要记住"^",如果没有它,那么这个block就真的成了花括号了。

2, 声明一个block
block的声明形式:
return_type (^block_name) (parameters)

举个例子:

int (^oneBlock) (NSInteger a, NSNumber *b);

在这里我们就声明了一个叫做oneBlock的block变量,它接受两个参数,并且返回值类型为int。

3,定义一个block

void (^clickBlock) (UIButton *btn);
clickBlock = ^(UIButton *btn) {
  NSLog(@"Button tag:%ld",btn.tag);
}

以上代码中,我们先声明了一个clickBlock,之后又对这个block进行了定义。

double (^ totalBlock) (double a , double b) = ^double (double a, double b) {
  return a + b;
}

以上代码我们在声明block的同时对它进行了定义。在这里声明部分的返回值double是可以省略不写的。因此可以写成这样:

double (^ totalBlock) (double a , double b) = ^(double a, double b) {
  return a + b;
}

4, 使用block
我们声明了block,定义了block,接下来当然是要使用它了。

int (^addBlock) (int a, int b) = ^(int a, int b) {
  return a + b;
}

int addition = addBlock (5, 10); //addition = 15

在这里,addBlock的使用方法就如同我们调用函数一样。传入参数,获得返回值。

除了以上这种简单的使用,下面要介绍的可以说是在使用block过程中的大杀器,内联块(inline block)。
以下的代码来自于我对于AFNetwroking的简单封装,大家可以从中看到内联块的使用方法。

#import <AFNetworking/AFNetworking.h>

/* Http Request Type */
typedef NS_ENUM(NSUInteger,HttpRequestType){
    HttpRequestTypeGet = 0,
    HttpRequestTypePost
};

/* Block for request success */
typedef void(^requestSuccess) (NSDictionary *object);

/* Block for request failed */
typedef void(^requestFailure)(NSError *error);

/* Block for upload progress */
typedef void(^uploadProgress)(float progress);

/* Block for dowload progress */
typedef void(^downloadProgress)(float progress);




@interface AFNetworkingTools : AFHTTPSessionManager

/* Create singleton instance */
+(instancetype) sharedManager;

+(void) requestWithType:(HttpRequestType) type withUrlString:(NSString *) urlStr withParameters:(id) parameters withSuccessBlock:(requestSuccess) successBlock withFailureBlock:(requestFailure) failureBlock progress:(downloadProgress) progress;


@end

从以上代码我们可以看到,首先我声明了四个block,之后我将block当作参数写进了我定义的类方法里。接下来让我们看看这个类方法是如何定义的。

+(void) requestWithType:(HttpRequestType)type withUrlString:(NSString *)urlStr withParameters:(id)parameters withSuccessBlock:(requestSuccess)successBlock withFailureBlock:(requestFailure)failureBlock progress:(downloadProgress)progress {
    
    switch (type) {
        case HttpRequestTypeGet: {
            [[AFNetworkingTools sharedManager] GET:urlStr parameters:parameters progress:^(NSProgress * _Nonnull downloadProgress) {
//                progress(downloadProgress.completedUnitCount / downloadProgress.totalUnitCount);
            } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
                successBlock(responseObject);
            } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
                failureBlock(error);
            }];
            break;
        }
        case HttpRequestTypePost: {
            [[AFNetworkingTools sharedManager] POST:urlStr parameters:parameters progress:^(NSProgress * _Nonnull uploadProgress) {
                progress(uploadProgress.completedUnitCount / uploadProgress.totalUnitCount);
            } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
                successBlock(responseObject);
            } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
                failureBlock(error);
            }];
            break;
        }
        default:
            break;
    }
}

在上面的代码中,我在自己定义的类方法中调用了作为参数传递的block,并且把相应的值作为参数写入了被调用的block中。
看到这些你可能有些迷惑,我们把block作为参数进行了传递,并且对这些block进行了调用,但是这些block并没有定义呀,那么这些block具体做了什么工作呢?别着急,block的定义马上就来。

[AFNetworkingTools requestWithType:HttpRequestTypeGet withUrlString:urlStr withParameters:nil withSuccessBlock:^(NSDictionary *object) {
        [self fetchData:object];
        
    } withFailureBlock:^(NSError *error) {
        NSLog(@"Error:%@",error.localizedDescription);
    } progress:nil];
    

看到这里是不是有些眼熟,没错。block的定义就在这里,当我们调用AFNetworkingTools的类方法时,我们会对当做参数传递进来的block进行定义。与此同时,我们也将获得block在被调用时传递过来的参数。也就是第二部分代码所传的参数。

看到这里你也许会感到疑惑,废了这么多功夫我们为什么要使用block呢?为什么不直接把(NSDictionary *object),(NSError *error)这些参数传递过来不就行了,何必再使用block呢?

我在刚学习block的时候也不明白这个道理,在这里我想说的是,其实AFNetworkingTools 继承了AFNetworking这个网络请求第三方库,因此我们在调用方法的时候进行了异步调用,所以我们并不能确定参数什么时候返回。因此,我们需要使用block作为参数返回时的桥梁去传递参数。因为当我定义的block被调用的时候,我们可以确定AFNetworking的异步调用已经结束,我们可以拿到正确的返回值。

从这里我们也可以看出,block是一个有效传递参数的方法,并且它降低了代码的耦合性。因为当我们更改block的定义的时候并不会改变block被调用时的实现。这也使得我们的代码更加灵活。

后记

在下一篇文章中,我将介绍block捕获变量时需要注意些什么,为什么有些变量能被block改变,有些则不行,为什么说可以把block当做对象看待,以及block的内部结构以及它在内存中的不同形式。如果你觉得这篇文章对你有用请点个赞吧,也希望你能关注我的简书。你的支持将是我不断提高自己的动力!

相关文章

  • Objective-C Block,从0到1(1)

    “block”? “块”? 相信有C语言编程基础的童鞋们对于block一定不会感到陌生,用咱们汉语说,block就...

  • Objective-C Block,从0到1(2)

    前言 在上一篇Objective-C Block,从0到1(1)中我已经对block的基本用法做了介绍,需要的朋友...

  • 专业书单

    1.《Effective Objective-C 2.0》288页 2.《Producter | 让产品从0到1》...

  • 从 0 到 1 认识从 0 到 1

    看了太多从 0 到 1 的标题了,总感觉那是在乱用流行的标题,记得这个标题是从阿里开始的,从 0 到 1 的书,活...

  • 从0到0,从0到1。

    昨天和一客户交流,听到这么一句话,我现在的阶段勉强算0到0的阶段,到那个1的阶段还没有看到,或者说并不知道那个1在...

  • 从1到0,0到1。

    把经历过的一切事情都归零是件很难可以做到的事情,并不会像计算器那么简单。 有时候想,如果人的大脑能像机器那样多好,...

  • 从0到1,从1到多。

    1、从0到1。学会做成一件事,改写人生。 POA: 最优秀榜样(作品)—价值观(目标)—持续的行动 学习的核心算法...

  • 从0到1

    【阅读感悟】 书名《书都不会读,你还想成功》 听了猫叔“更好的表达课”,发现自己太差劲了,做什么事都是三分热度,不...

  • 从0到1

    1第一层境界:企业只是制造满足市场需求的产品,只要有原型,工业流水线可以让产品大量地复制生产出来。但产品有生命周期...

  • 从0到1

    1创 造性垄断就是新产品既让大众受益,又可以给创造者带来长期利润。竞争意味着大家都没有利润,产品没有实质差异,而且...

网友评论

    本文标题:Objective-C Block,从0到1(1)

    本文链接:https://www.haomeiwen.com/subject/pdaajttx.html