浅谈block和delegate的使用

2021-04-19 11:00:23

浅谈block和delegate的使用

delegate

委托是协议的一种,顾名思义,就是委托他人帮自己去做事。委托是给一个对象提供机会对另一个对象中的变化做出反应或者影响另一个对象的行为。其基本思想是:两个对象协同解决问题,并且打算在广泛的情形中重用。委托指向另一个对象(即它的委托)的引用,并在关键时刻给委托发消息。消息可能只是通知委托发生了某件事情,给委托提供机会执行额外的处理,或者消息可能要求委托提供一些关键的信息以控制所发生的事情。委托的作用主要有两个,一个是传值,一个是传事件。

传值常用在B类要把自己的一个数据或者对象传给A类,让A类去展示或者处理(切分紧耦合,和代码分块时常用)。传事件是A类发生了什么事,把这件事告诉关注自己的类,也就是委托的对象,由委托的对象去考虑发生这个事件后应该做出什么反映(例如在异步请求中,界面事件触发数据层改变等)。利用委托赋值,这种方法是为了不暴露自己的属性就可以给自己赋值,这样方便了类的管理,只有在你想要让别人给你赋值的时候才调用,这样的赋值更可控一些。(如tableView中的委托dateSource等)。

在iOS中委托通过一种@protocol的方式实现,所以又称为协议。协议是多个类共享的一个方法列表,在协议中所列出的方法没有响应的实现,由其它类来实现。delegate是“一对一”的关系,对同一个协议,一个对象只能设置一个代理delegate,所以单例对象就不能用代理。代理更注重过程信息的传输:如发起一个网络请求,是否此时请求已经开始、是否收到了数据、数据是否已经接受完成、数据接收失败等。

从委托类的定义可以看出,委托与协议有一定的相似性。都采用protocol关键字来声明,并且其中的方法都有optional和required,都需要对required方法和调用的optional方法进行实现。而不同的是在委托对象所在的类中需要定义一个delegate对象,并且为id类型。但是delegate与protocol本质上是不同的。Delegate本身应该称为一种设计模式,是把一个类自己需要做的一部分事情,让另一个类(也可以就是自己本身)来完成,而实际做事的类为delegate。而protocol是一种语法,它的主要目标是提供接口给遵守协议的类使用,而这种方式提供了一个很方便的、实现delegate模式的机会。

委托模式的实现思路:

1、通常是在对象主体包含一个委托对象的弱引用:

@property (nonatomic, weak, nullable) id <UITableViewDelegate> delegate;

2、委托对象的实现有两种方式:一种是必须实现,一种是可选实现,即@required和@optional的区别。

@protocol UITableViewDelegate<NSObject, UIScrollViewDelegate>@optional- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath;

3、判断触发委托方法。

//确定委托是否存在Entered方法if([delegate respondsToSelector:@selector(method:)]){
    //发送委托方法,方法的参数为用户的输入
    [delegate method:xxx];}

4、连接对象主体和委托,通过setDelegate:(id)obj来实现。

_test.delegate = self;

5、如果实现委托的类有委托需要的方法就执行方法。大概就是这样。

Block

Block是Apple Inc.为C、C++以及Objective-C添加的特性,使得这些语言可以用类lambda表达式的语法来创建闭包,是iOS4.0以后和Mac OS X 10.6以后引进的对C语言的扩展,用来实现匿名函数的特性。 Block能够读取其它函数内部变量的函数,在一段请求连续代码中可以看到调用参数(如发送请求)和响应结果。采用Block技术能够抽象出很多共用函数,提高了代码的可读性,可维护性,封装性。

block写法更简练,不需要写protocol、函数等等;block注重结果的传输:比如对于一个事件,只想知道成功或者失败,并不需要知道进行了多少或者额外的一些信息;block需要注意防止循环引用。

block的写法大概就是这样:

void (^blockTest)(void) = ^{
    NSLog(@"block");};blockTest();

如果用block进行两个类之间的互动,需要这样:
BLOCK首先你在.h文件中声明BLOCK对象,当然返回的参数你可以自己定义:

typedef void (^blockTest)(); @property (nonatomic,copy) blockTest myblock;

在.m文件中执行时:

if(_myblock){
_myblock();
}

其他类执行block时是这样的:

_testClass.myblock = ^{
 NSLog(@"block");}

需要注意的是:
1.循环引用问题,如果block里用了self,需要替换为

__weak typeof(self) weakSelf = self;

2.block所在函数中的,捕获自动变量,但是不能修改它,不然就是编译错误。但是可以改变全局变量、静态变量、全局静态变量。为何不让修改变量:这个是编译器决定的。理论上当然可以修改变量了,只不过block捕获的是自动变量的副本,名字一样。为了不给开发者迷惑,干脆不让赋值。静态变量属于类的,不是某一个变量。所以block内部不用调用self指针,所以block可以调用。解决block不能保存值这一问题的另外一个办法是使用__block修饰符。

block和delegate的区别

block不像代理声明了一个代理函数,在调用的类内部还要实现该函数,若一个页面能发送多个请求,并且用多点触控同时触发发送多个请求,那个这个页面的代理函数很难区分是那个请求的结果,只有你的响应消息中带有消息类型可能会分出来,若服务器做的不够强大,当出现异常时,找不发送请求,对于开发来说是个问题。这样多个消息在一个函数里解析也不利于封装。 Block比代理更清晰, Block可以在创建事件的时候区分开来。这也是为什么现在苹果 API 中越来越多地使用 Block而不是 Delegate。

block有三个存储区域_NSConcretStackBlock ,_NSConcretGlobalBlock和_NSConcretMallocBlock。正如它们名字说的那样,说明了block的三种存储方式:栈、全局、堆。如果是定义在函数外面的block是global的,另外如果函数内部的block但是没有捕获任何自动变量,那么它也是全局的。

typedef  int  (^blk_test)( int );for(...){blk_test blk = ^(int count) {return count;};}

虽然,这个block在循环内,但是blk的地址总是不变的。说明这个block在全局段。在栈上block调用copy那么复制到堆上,在全局block调用copy什么也不做,在堆上调用block引用计数增加。

测试方法如下:

typedef  int  (^blkt1)(void) ;-(void) stackOrHeap{__block int val =10;int *valPtr = &val;//使用int的指针,来检测block到底在栈上,还是堆上blkt1 s= ^{

    NSLog(@"val_block = %d",++val);

    return val;};s();NSLog(@"valPointer = %d",*valPtr);}

int val肯定是在栈上的,我保存了val的地址,看看block调用前后是否变化。输出一致说明是栈上,不一致说明是堆上。有兴趣的可以手动试一下ARC和MRC下的结果。

由此我们可以看到delegate运行成本低,block成本很高。block出栈需要将使用的数据从栈内存拷贝到堆内存,当然对象的话就是加计数,使用完或者block置nil后才消除;delegate只是保存了一个对象指针,直接回调,没有额外消耗。相对C的函数指针,只多做了一个查表动作 。

一般来说公共接口,方法也较多用delegate进行解耦 ,iOS有很多例子如最常用tableViewDelegate,textViewDelegate等。异步和简单的回调用block更好 ,iOS有很多例子如常用的网络库AFNetwork等。



作者:一个人在路上走下去
链接:https://www.jianshu.com/p/bbccac805641
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


  • 2019-10-09 14:39:40

    import双反斜杠\\的意思

    ​ \表示引用根目录下面的PHPEXcel;不用\的话是引用当前目录下面的 PHPExcel

  • 2019-10-09 15:33:31

    nuxt,nuxtjs简单介绍以及使用

    在集成的服务器端框架之间进行选择: 选择您喜欢的 UI 框架: 选择您喜欢的测试框架: 选择你想要的 Nuxt 模式 (Universal or SPA) 添加 axios module 以轻松地将 HTTP 请求发送到您的应用程序中。 添加 EsLint 以在保存时代码规范和错误检查您的代码。 添加 Prettier 以在保存时格式化/美化您的代码。

  • 2019-10-10 00:21:35

    laravel 5.6以上日志理解及日志格式定义

    Laravel/Lumen的日志默认是基于Monolog进行了一层封装,如果要求不高,用起来还是十分容易的,本文基于laravel5.6/Lumen5.6版本进行解说。5.6版对日志系统做了升级,将日志的配置单独放以了config/logging.php 配置文件中,所以现在实用多了。

  • 2019-10-10 10:10:49

    @Scheduled注解各参数详解

    每隔5秒执行一次:*/5 * * * * ? 每隔1分钟执行一次:0 */1 * * * ? 每天23点执行一次:0 0 23 * * ? 每天凌晨1点执行一次:0 0 1 * * ? 每月1号凌晨1点执行一次:0 0 1 1 * ? 每月最后一天23点执行一次:0 0 23 L * ? 每周星期天凌晨1点实行一次:0 0 1 ? * L 在26分、29分、33分执行一次:0 26,29,33 * * * ? 每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?

  • 2019-10-10 11:07:47

    Java8 - Map更优雅的迭代方式:forEach

    用于两个参数之间进行操作的函数式接口是 BiConsumer。这个函数式接口正好用来操作 Map 的 key 和 value。JDK8强化了针对 Map 类的迭代方式,新增了一个默认方法 forEach,它接收一个 BiConsumer 函数。JDK给出的描述如下:

  • 2019-10-11 13:48:00

    关于Integer比较相等的问题

    原来两个Integer类型的数字不能用==来判断,要用equal 不过Integer类型的可以与1,2这样的纯数字来判断