【我就看看不说话】Block 使用

来源:互联网 发布:mac mini 2016 10月 编辑:程序博客网 时间:2024/04/25 17:30


摘要 Block 与传统代码相比较更加轻量,调用简洁方便,而且可以返回多个参数,使用Block可以让代码更加具有易读性,而我们在写回调时,也可以直接写在函数内部,而不用再去写一个回调函数

Block闭包 iOS Objective-C


目录[-]

Blocks语法

Blocks省略句式

省略返回值类型

省略参数列表

Block类型变量

简单用法

回调

传值

BlocksC语言的扩充功能:带有自动变量(局部变量)的匿名函数。通过Blocks,源代码中就能使用匿名函数,即不带名称的函数。在我们的工作中,命名占据了很大一部分,函数名,变量名,属性名,类名,框架名等都必须具备。能够编写不带名称的函数对程序员来说是具有相当吸引力的。


Blocks语法


完整形式的Blocks 与一般的C语言函数相比较,有两点不同


没有函数名

带有 ^

Blocks BN范式


Block_literal_expression ::= ^ block_decl compound_statement_body

block_decl ::=

block_decl ::= parameter_list

block_decl ::= type_expression

翻译成大白话就是


^ 返回值类型 (参数列表)表达式


1.返回值类型 OC/C语法中的返回值类型


2.参数列表 C语法中的参数列表,OC中的参数是一个个传的,这里的语法更像 C语言中以()包含的参数列表

3.表达式 OC/C语法中允许使用的表达式



^int (int count){return count++; }


^NSString * (NSNumber *num){

   return [NSString stringWithFormat:@"%@",num];

};

大家可能会疑惑,为什么平时看到的 Blocks 并不全是如此的,因为Blocks可以省略好几个项目


Blocks省略句式


省略返回值类型


如:


^ 返回值类型 (参数列表)表达式


可将返回值类型省略


上面的代码例子可写为


^(int count){return count++; }


^(NSNumber *num){

   return [NSString stringWithFormat:@"%@",num];

};

如此,看起来是不是熟悉多了。


省略返回值类型时,如果表达式中有return语句就使用该返回值的类型,如果表达式中没有return语句就使用void类型,见下文代码例子。


下面给出完整定义句式(使用省略返回值类型句式)


^(int count){return count+1 ;};


^(NSNumber *num){

   return [NSString stringWithFormat:@"%@",num];

};


^(NSNumber *count){

    NSLog(@"无返回类型Block count = %@", count);

};

^(void){

    NSLog(@"无返回类型也无参数 Block");

};

省略参数列表


如果不使用参数,参数列表也可省略


^ 返回值类型 (参数列表)表达式


上文无返回值类型,也无参数的Block也简化为


^{

    NSLog(@"无返回类型也无参数 Block");

};

Block类型变量


上面所讲述的Block从语法格式上来看与除了无名称及带有 ^ 之外,与 C/OC定义相同。


Block语法下,可将 Block 语法赋值给声明为 Block类型的变量中。 即源代码中一旦使用 Block语法就相当于生成了可赋值给 Block 类型变量的 Blocks中由 Block语法生成的值也被称为Block


声明 Block 类型变量的示例如下:


int (^counts)(int);

Block 类型变量与其他 C/OC变量没有任何区别,可以作为以下用途使用


自动变量

函数参数

静态变量

静态全局变量

全局变量

下面我们试着 Block 语法将 Block 赋值给 Block 类型变量


int (^counts)(int) = ^(int count){return count+1 ;};


NSString *(^str)(NSNumber *num) = ^NSString *(NSNumber *num){

    return [NSStringstringWithFormat:@"%@",num];

};


void (^blank)(NSNumber *count) = ^(NSNumber *count){

    NSLog(@"无返回类型Block count = %@", count);

};

void (^blank)(void) = ^(void){

    NSLog(@"无返回类型也无参数 Block");

};

在函数参数中使用 Block 类型变量向函数 传递Block,即 Block 变量作为函数的形参


void func(int (^counts)(int)){

}

Block 作为类型变量传参时,记述方式及其复杂,如上文形参,这是我们可以像使用函数指针类型时那样,使用 typedef来简化记述方式。


typedefint (^count) (int);

重写上面的方法


void func(count num){

    

}

这是 C 语言的写法,换成 OC写法,让大家更加清楚一点,在定义时也是如此使用


/**

 * 原来的写法

 */

-(void)funcWithCount:(int (^)(int))count{

    

}


/**

 * 使用typedef之后的写法

 */

-(void)funcWithCount:(count )count{

    

}

简单用法


定义了 Block代码块之后,就可以将一整块代码当做一个变量来使用,变量可为局部变量,也可为全局变量,这也是我认为 Block最方便之处。


假设要生成两个数组,一个装有5个随机数,一个装有10个随机数,将生成随机数的方法定义为一个闭包,在后文既可直接访问,如


NSNumber *(^randArray)(void) = ^{

   int rand = arc4random() % 100;

    

    NSNumber *number = [NSNumber numberWithInt:rand];

   return number;

};


NSMutableArray *array1 = [[NSMutableArrayalloc] init];

NSMutableArray *array2 = [[NSMutableArrayalloc] init];


for (NSInteger index =0; index<10; index++) {

    [array1 addObject:randArray()];

}


for (NSInteger index =0; index<5; index++) {

    [array2 addObject:randArray()];

}

回调


下面给出一 tableViewCell 上按钮事件回调的例子,这个也是很多人头痛的问题,通过block可以很方便地实现,而且层次非常清晰。


自定义cell命名为blockCellcell上放一switch 控件,我们希望switch被点击时在viewController中可以得到switch的状态,获取到点击事件。


blockCell.h中定义一Block


typedefvoid(^switchAction)(blockCell *);

@property (nonatomic,copy)switchAction action;

switch的点击时间事件中调用switchAction


blockCell.m


- (IBAction)switchToggle:(id)sender {

   self.action(self);

}

viewController中使用这个自定义Celltable进行初始化


-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{

    

    NSString *cellID =@"blockCell";

    

    blockCell *cell = [tableView dequeueReusableCellWithIdentifier:cellID];

    

    cell.action = ^(blockCell *cell){

        // 既可取到行,也可取到switch状态

        NSLog(@"行数:%ld, switch state: %u", (long)indexPath.row, cell.switchBtn.on);

        

    };

    

   return cell;

}

传值


现在很多流行地第三方库都将回调改成了Block,之前用的Delegate特别得心应手有木有,都封装好了直接调用得到我要的结果,好了,都改成Block,不知道如何去接Block的返回值,只能一遍又一般地重写。


其实要封装很容易,将第三方库返回的Block,以一个Block来接住再返回调用的页面就可以了,本想介绍

AFNetworing后再讲这个,但是我看了下,github上他们的主页的readMe写得超级清楚详细,想要了解的童鞋请仔细看下他们的readMe


添加方式


Github地址:https://github.com/AFNetworking/AFNetworking


可以将类库拷贝到工程目录下添加,推荐用 cocoapods 安装,方便更新,而且不用手动导入framework,一键设置


封装


目的:将参数传递后调用对应的方法直接得到网络返回值


新建一个类WebRequest ,此处写一个示例,大家自己参考


#import <Foundation/Foundation.h>


#import "AFNetworking.h"


@interface WebRequest :NSObject



-(void)requestNameWithID:(NSString *)ID

             WithSuccess:(void (^)(AFHTTPRequestOperation *operation,id responseObject, NSDictionary *myData))success

                 failure:(void (^)(AFHTTPRequestOperation *operation,NSError *error))failure;


@end

@implementation WebRequest


-(void)requestNameWithID:(NSString *)ID

             WithSuccess:(void (^)(AFHTTPRequestOperation *operation,id responseObject, NSDictionary *myData))succes

                 failure:(void (^)(AFHTTPRequestOperation *operation,NSError *error))failure

{

    

    

   NSURL *url = [NSURLURLWithString:@"ID拼接地接口地址"];

   NSURLRequest *request = [NSURLRequestrequestWithURL:url];

    AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc] initWithRequest:request];

    [operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *operation,id responseObject) {

        

        NSDictionary *dic = responseObject[@"someKey"];

        

        success(operation, responseObject, dic);//此处将网络返回值传递给我们自己定义的Block中的三个返回值,dic可以自定义,也可不加,如此可以返回经自己在这里已经处理好的对象,而不必调用一次,处理一次

        

    } failure:^(AFHTTPRequestOperation *operation, NSError *error) {

        failure(operation,error);//与方法定义中的Block一致

    }];

    

}



@end

调用


WebRequest *request = [[WebRequest alloc] init];


[request requestNameWithID:@"123"

               WithSuccess:^(AFHTTPRequestOperation *operation,id responseObject, NSDictionary *myData) {

                   

                   // 在网络成功时在这里就可以得到返回值了,operationresponseObjectmyData

                   

                   

               }

                   failure:^(AFHTTPRequestOperation *operation, NSError *error) {

                       

                      // 网络失败回调

                       

                   }];



示例工程下载




芳仔说:


现在越来越多地库使用Block^作为Block的标志,初看会很不适应,而且在未使用的情况下会对其有抵触心理


iOS8也有相当多的Block操作,Block出现已经有两年多了,正如日中天,取代delegate也不远了,相信大家再稍微探究使用后会爱上它的,祝大家好运!


未免篇幅太长,新手看蒙圈,会在后续文章中写关于 Block 对象的进阶用法,基本用法看过本文之后就可以很得心应手了



0 0
原创粉丝点击