Block Delegate and Closure

1. About Block Format


As a local variable:


returnType (^blockName)(parameterTypes) = ^returnType(parameters) {...};

As a property:


@property (nonatomic, copy, nullability) returnType (^blockName)(parameterTypes);

As a method parameter:


- (void)someMethodThatTakesABlock:(returnType (^nullability)(parameterTypes))blockName;

As an argument to a method call:


[someObject someMethodThatTakesABlock:^returnType (parameters) {...}];

As a typedef:


typedef returnType (^TypeName)(parameterTypes);

TypeName blockName = ^returnType(parameters) {...};



2. About the relationship between Block and Delegate


https://onevcat.com/2011/11/objc-block/


// Defining a block variable

BOOL (^isInputEven)(int) = ^(int input) {

    if (input % 2 == 0) {

        return YES;

    } else {

        return NO;

    }

};

以上定义了一个block变量,block本身就是一个程序段,因此有返回值有输入参数,这里这个block返回的类型为BOOL。天赋异秉的OC用了同样不走寻常路的" "符号来表示block定义的开始(就像用减号和加号来定义方法一样),block的名称紧跟在 符号之后,这里是isInputEven(也即以后使用inline方式调用该block时所需要的名称)。这段block接受一个int型的参数,而在等号后面的int input是对这个传入int参数的说明:在该block内,将使用input这个名字来指代传入的int参数。一开始看block的定义和写法时可能会比较痛苦,但是请谨记它只是把我们常见的方法实现换了一种写法而已,请以习惯OC中括号发送消息的速度和决心,尽快习惯block的写法吧!

调用这个block的方法就非常简单和直观了,类似调用c函数的方式即可:


// Call similar to a C function call

int x = -101;

NSLog(@"%d %@ number", x, isInputEven(x) ? @"is an even" : @"is not an even");


https://my.oschina.net/leejan97/blog/209762


代理设计模式对于iOS开发的人来说肯定很熟悉了,代理delegate就是委托另一个对象来帮忙完成一件事情,为什么要委托别人来做呢,这其实是MVC设计模式中的模块分工问题,例如View对象它只负责显示界面,而不需要进行数据的管理,数据的管理和逻辑是Controller的责任,所以此时View就应该将这个功能委托给Controller去实现,当然你作为码农强行让View处理数据逻辑的任务,也不是不行,只是这就违背了MVC设计模式,项目小还好,随着功能的扩展,我们就会发现越写越难写;还有一种情况,就是这件事情做不到,只能委托给其他对象来做了,下面的例子中我会说明这种情况。


下面的代码我想实现一个简单的功能,场景描述如下:TableView上面有多个CustomTableViewCell,cell上面显示的是文字信息和一个详情Button,点击button以后push到一个新的页面。为什么说这个场景用到了代理delegate?因为button是在自定义的CustomTableViewCell上面,而cell没有能力实现push的功能,因为push到新页面的代码是这样的,


[self.navigationController pushViewController...];


所以这时候CustomTableViewCell就要委托它所在的Controller去做这件事情了。


按照我的编码习惯,我喜欢把委托的协议写在提出委托申请的类的头文件里面,现在的场景中是CustomTableViewCell提出了委托申请,下面是简单的代码,


@protocol CustomCellDelegate <NSObject>


- (void)pushToNewPage;


@end




@interface CustomTableViewCell : UITableViewCell


@property(nonatomic, assign) id<CustomCellDelegate> delegate;



@property (nonatomic, strong) UILabel *text1Label;


@property(nonatomic,strong) UIButton *detailBtn;


@end


上面的代码在CustomTableViewCell.h中定义了一个协议CustomCellDelegate,它有一个需要实现的pushToNewPage方法,然后还要写一个属性修饰符为assign、名为delegate的property,之所以使用assign是因为这涉及到内存管理的东西,以后的博客中我会专门说明原因。


接下来在CustomTableViewCell.m中编写Button点击代码,



[self.detailBtn addTarget:self action:@selector(btnClicked:) forControlEvents:UIControlEventTouchUpInside];


对应的btnClicked方法如下,


- (void)btnClicked:(UIButton *)btn


{


    if (self.delegate && [self.delegaterespondsToSelector:@selector(pushToNewPage)]) {


        [self.delegate pushToNewPage];


    }


}


上面代码中的判断条件最好是写上,因为这是判断self.delegate是否为空,以及实现CustomCellDelegate协议的Controller是否也实现了其中的pushToNewPage方法。


接下来就是受到委托申请的类,这里是对应CustomTableViewCell所在的ViewController,它首先要实现CustomCellDelegate协议,然后要实现其中的pushToNewPage方法,还有一点不能忘记的就是要设置CustomTableViewCell对象cell的delegate等于self,很多情况下可能忘了写cell.delegate = self;导致遇到问题不知云里雾里。下面的关键代码都是在ViewController.m中,


首先是服从CumtomCellDelegate协议,这个大家肯定都知道,就像很多系统的协议,例如UIAlertViewDelegate、UITextFieldDelegate、UITableViewDelegate、UITableViewDatasource一样。


@interface ViewController ()<CustomCellDelegate>


@property (nonatomic, strong) NSArray *textArray;




@end


然后是实现CustomCellDelegate协议中的pushToNewPage方法,


- (void)pushToNewPage


{


    DetailViewController*detailVC = [[DetailViewController alloc] init];


    [self.navigationController pushViewController:detailVC animated:YES];


}


还有一个步骤最容易被忘记,就是设置CumtomTableViewCell对象cell的delegate,如下代码,


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


{


    static NSString *simpleIdentify = @"CustomCellIdentify";


    CustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:simpleIdentify];


    if (cell == nil) {


        cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:simpleIdentify];


    }


    //下面代码很关键


    cell.delegate = self;


    cell.text1Label.text = [self.textArray objectAtIndex:indexPath.row];


    return cell;


}


通过cell.delegate = self;确保了CustomTableViewCell.m的判断语句if(self.delegate && ...){}中得self.delegate不为空,此时的self.delegate其实就是ViewController,cell对象委托了ViewController实现pushToNewPage方法。这个简单的场景描述了使用代理的一种情况,就是CustomTableViewCell没有能力实现pushViewController的功能,所以委托ViewController来实现。


代码在github可以下载。


有什么错误,还请大家指正。


----------------------------------------------下面是block的内容----------------------------------------------------


Block是一个C语言的特性,就像群里有人说的,它就是C语言的函数指针,在使用中最多的就是进行函数回调或者事件传递,比如发送数据到服务器,等待服务器反馈是成功还是失败,此时block就派上用场了,这个功能的实现也可用使用代理,这么说的话,感觉block是不是有点像代理了呢?


我之前接触block,都是使用它作为函数参数,当时感觉不是很理解。现在在项目中,很多时候block作为property,这样更加简单直接,想想,其实property不就是定义的合成存储的变量嘛,而block作为函数参数也是定义的变量,所以作为函数参数或者作为property本质没有区别。


看一看别人总结的block的语法吧,http://fuckingblocksyntax.com,这个链接亮了,fucking block syntax,操蛋的block语法啊。block有如下几种使用情况,


1、作为一个本地变量(local variable)


returnType (^blockName)(parameterTypes) = ^returnType(parameters) {...};


2、作为@property


@property (nonatomic, copy) returnType (^blockName)(parameterTypes);


3、作为方法的参数(method parameter)


- (void)someMethodThatTakesABlock:(returnType (^)(parameterTypes))blockName;


4、作为方法参数的时候被调用


[someObject someMethodThatTakesABlock: ^returnType (parameters) {...}];


5、使用typedef来定义block,可以事半功倍


typedef returnType (^TypeName)(parameterTypes);

TypeName blockName = ^returnType(parameters) {...};


上面我也只是复制粘贴了一下,接下来还是实现点击CustomTableViewCell上面的Button实现页面跳转的功能,我之前不止一次的类比block就像delegate,这边我也是思维惯性,下面的内容我就当block为代理,一些用词描述还是跟delegate差不多。首先,在提出委托申请的CustomTableViewCell中定义block的property,


@interface CustomTableViewCell : UITableViewCell


@property (nonatomic, strong) UILabel *text1Label;


@property (nonatomic, strong) UIButton *detailBtn;




//下面的定义,请看官们对比一下


/*delegate的定义 我没有删除,因为大家可以类比了看下*/


@property (nonatomic, assign) id<CustomCellDelegate> delegate;


/*这里定义了ButtonBlock*/


@property (nonatomic, copy) void (^ButtonBlock)();


@end


这里用copy属性来修饰ButtonBlock property,这个原因,我会在以后的博客中作专门的解释。


接下来在CustomTableViewCell中给它上面的detailBtn绑定点击方法,


[self.detailBtn addTarget:self action:@selector(btnClicked:) forControlEvents:UIControlEventTouchUpInside];


然后是btnClicked方法的细节,我把delegate的内容也没有删除,就是给各位比较一下block和delegate的功能和语法的相似性,


- (void)btnClicked:(UIButton *)btn


{


    //这是之前的delegate


    if (self.delegate && [self.delegate respondsToSelector:@selector(pushToNewPage)]) {


        [self.delegate pushToNewPage];


    }


    


    //这是现在我们要说的block    


    if (ButtonBlock) {


        ButtonBlock();


    }


}


下面是一个关键性的地方,在ViewController2中设置其CustomTableViewCell的cell对象的ButtonBlock,也就是给它赋值,此处我还是保留了cell.delegate = self;代码,


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


{


    NSString *blockIdentify = @"BlockIdentify";


    CustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:blockIdentify];


    if (cell == nil) {


        cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:blockIdentify];


    }


    cell.text1Label.text = [self.textArray objectAtIndex:indexPath.row];


    //delegate的不可缺少的代码,这里放在这儿只是为了给各位类比一下


    cell.delegate = self;


    


    //ButtonBlock不可缺少的代码


    cell.ButtonBlock = ^{


        [self pushToNewPage2];


    };


    return cell;


}


之所以cell.ButtonBlock = ^{};赋值,是因为我们我们是这样定义ButtonBlock的,void (^ButtonBLock)(),表示无返回值无参数。


然后编写pushToNewPage2方法,


- (void)pushToNewPage2


{


    DetailViewController *detailVC = [[DetailViewController alloc] init];


    [self.navigationController pushViewController:detailVC animated:YES];


}


你们看这个方法是不是与CustomCellDelegate协议中的pushToNewPage方法类似。然后在回过头来类比一样,是不是block就是精简版的delegate,因为delegate设计模式要写协议CustomCellDelegate,还有容易遗漏cell.delegate = self;但是block使用的时候就简单多了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值