Block很简单,就像delegate的简化版

代理设计模式对于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使用的时候就简单多了。

最新的代码我已经更新到github,非常简单,有疑问或者有质疑的朋友可以下载看看。

另:本人也是ios开发菜鸟,对于很多的细节了解不够深入,上面的代码使用arc编码,如果我写的有什么问题或者有什么缺陷的话,还请大神给我指正。

很希望有各位开发者相互交流,提高技术,我的邮箱:[email protected]。还有一个qq交流群188647173,有兴趣的可以加进来讨论讨论、学习学习,也希望牛逼的大神到群里给我们菜鸟指点指点。

时间: 2024-10-23 12:10:01

Block很简单,就像delegate的简化版的相关文章

Block就像delegate的简化版

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

手把手教你开发chrome扩展一:开发Chrome Extenstion其实很简单

手把手教你开发chrome扩展一:开发Chrome Extenstion其实很简单 手把手教你开发chrome扩展一:开发Chrome Extenstion其实很简单 手把手教你开发Chrome扩展二:为html添加行为 手把手教你开发Chrome扩展三:关于本地存储数据 Chrome的更新速度可以说前无古人,现在我每天开机的第一件事就是打开Chrome检查是不是有了新版本.界面清爽.操作人性化.网络备份资料和快速的启动速度令我爱不释手,还有它拥有众多的扩展程序,相对于firefox的插件来说,

Java断点续传实现原理很简单

原理解析 在开发当中,"断点续传"这种功能很实用和常见,听上去也是比较有"逼格"的感觉.所以通常我们都有兴趣去研究研究这种功能是如何实现的? 以Java来说,网络上也能找到不少关于实现类似功能的资料.但是呢,大多数都是举个Demo然后贴出源码,真正对其实现原理有详细的说明很少. 于是我们在最初接触的时候,很可能就是直接Crtl + C/V代码,然后捣鼓捣鼓,然而最终也能把效果弄出来.但初学时这样做其实很显然是有好有坏的. 好处在于,源码很多,解释很少:如果我们肯下功

很简单的JQuery网页换肤

现在介绍网页换肤的例子已经数不胜数,自己也没有仔细研究过,刚搞遇到类似这个问题,网上查了资料解决了问题,感觉很有用,就记录了下来和大家分享一下.效果很简单,我就直接讲怎么使用的. 其中用到了jquery.cookie.js 首先是html代码: <div id="header_demo"> <a id="logo" href="#">Rainweb</a> <ul id="skin"&

iOS Block的简单使用

目录:本文将分三个部分讲解block块的基本用法,由于本人水平有限,不足之处还望指正,谢谢! 一.block块的基本用法: 相信大家都很熟悉了,下面直接上代码: //1,block初探,无参数,无返回值 void (^MyBlock)(void) = ^(void){ NSLog(@"2,打印出来了,我就是block,无参数,无返回值"); }; NSLog(@"1"); MyBlock(); //2,有参数,无返回值 void (^MyblockTwo)(int

log4j入门(转) --- 很详细 也很简单容易懂

log4j入门(转) Log4j实在是很熟悉,几乎所有的Java项目都用它啊.但是我确一直没有搞明白.终于有一天我受不了了,定下心去看了一把文档,才两个小时,我终于搞明白了.一般情况下Log4j总是和Apache Commons-logging一起用的,我也就一起介绍吧.多了个东西不是更麻烦,而是更简单! 在Log4J中存在几个概念首先介绍一下,最最重要的两个是Logger和Appender(请参考Log4J手册),其实是继承层次以及输出控制. 首先Log4J中总是存在一个rootLogger,

很简单的Java断点续传实现原理

原理解析 在开发当中,"断点续传"这种功能很实用和常见,听上去也是比较有"逼格"的感觉.所以通常我们都有兴趣去研究研究这种功能是如何实现的? 以Java来说,网络上也能找到不少关于实现类似功能的资料.但是呢,大多数都是举个Demo然后贴出源码,真正对其实现原理有详细的说明很少. 于是我们在最初接触的时候,很可能就是直接Crtl + C/V代码,然后捣鼓捣鼓,然而最终也能把效果弄出来.但初学时这样做其实很显然是有好有坏的. 好处在于,源码很多,解释很少:如果我们肯下功

王金战:改变一个差生真的很简单

当一个学生,反复遭遇失败的打击,他就变成了差生,没有一个学生生下来就注定是个差生.所以让一个差生变好真的很简单.    我刚参加工作的时候,不敢说这句话,但是我现在敢说,因为我多年的经历已经证明了这件事情.让一个差生变好真的很简单,怎么做呢?就是反其道而行之.差生是反复遭遇失败的打击后才产生的,让一个差生变好,就是让他反复享受到成功的喜悦,这个学生就会慢慢地变好了.    少年来我就用这样的方法和理念,帮着一个个学生走出困境,走向成功.我深深体会到,作为教师,作为家长,要学会欣赏孩子.特别是对那

【结果很简单,过程很艰辛】记阿里云Ons消息队列服务填坑过程

Maybe 这个问题很简单,因为解决方法是非常简单,但填坑过程会把人逼疯,在阿里云ONS工作人员.同事和朋友的协助下,经过一天的调试和瞎捣鼓,终于解决了这个坑,把问题记下来,也许更多人在碰到类似问题的时候,会开放思路.当然不得不说,Ons的.NET接口还很不完善,甚至没有独立在Windos 2008/2012服务器测试过,希望官方加把力. 1.阿里云ONS介绍 ONS(Open Notification Service)即开放消息服务,是基于阿里开源消息中间件MetaQ(RocketMQ)打造的