delegate和protocol
protocol和delegate完全不是一回事,放在一起说,只是因为我们经常在同一个头文件里看到这两个word。
协议(protocol),就是使用了这个协议后就要按照这个协议来办事,协议要求实现的方法就一定要实现。
委托(delegate),顾名思义就是委托别人办事,就是当 一件事情发生后,自己不处理,让别人来处理。
举个浅显的例子:
我上班的工作主要内容包括 (1)写代码(2)写文档(3)测试程序(4)接电话(5)会见客户
(1)(2)我自己全权负责,但是后面(3)(4)(5)我不想或者不方便自己做,所以我想找个助手(delegate)帮我做这些事,于是我定了一个招聘要求(Protocol),里写明我的助手需要会做(3)(4)(5)这三件事。很快,我招到一个助手。
即:我.delegate = 助手;
于是以后每当我遇到需要测试程序或者接电话的活,我就把他转交给助手(delegate)去处理,助手处理完后如果有处理结果(返回值)助手会告诉我,也许我会拿来用。如果不需要或者没有结果,我就接着做下面的事。。
protocol和java里interface的概念类似,是Objective-C语法的一部分。
定义protocol如下
@protocol ClassADelegate- (void)methodA; - (void)methodB;@end
那么就是定义了一组函数,这组函数放在一起叫作一个protocol,也就是协议。
函数是需要被实现的,所以如果对于class如下
@interface ClassB <ClassADelegate> { } @end
就叫作ClassB conform to protocol ClassADelegate,也就是说ClassB实现了这个协议,
也就是实现了这一组函数。
有了上面这个头文件,我们就可以放心作调用
ClassB *b = [[ClassB alloc] init]; [b methodA]; [b methodB];
而不用担心出现unrecognized selector sent to instance这种错误了。
所以protocol就是一组函数定义,是从类声明中剥离出来的一组定义。
id<ClassADelegate> b = ...; [b methodA];
这种用法也常见,b是一个id类型,它知道ClassADelegate这组函数的实现。
那么delegate是什么?其实和protocol没有关系。Delegate本身应该称为一种设计模式。
是把一个类自己需要做的一部分事情,让另一个类(也可以就是自己本身)来完成。
比如ClassC
@interface ClassC {id delegate; } @end
那么ClassC的实现(.m文件)里就可以用delegate这个变量了。
当然这里完全可以用其它名字而不是delegate。
我们也可以这样写
@interface ClassC {ClassB *delegate; } @end
这样我们知道了delegate是一个ClassB,它就可以提供ClassB里的方法。
可以把一部分ClassC里的工作放在ClassB里去实现。
这样的写法看起来是不是有点奇怪?或者应该写成这样?
@interface ClassC {ClassB *classB; } @end …
delegate没有了…
所以说其实delegate只是一种模式,大家约定俗成,当把自己内部一部分实现暴露给另外一个类去做的时候,就叫实际做事的类为delegate。
为什么会需要把内部实现提出来给另一个类做呢?
最常见的目的就是为了在隐藏实现的前提下,提供一个自定义的机会。
比如Apple提供的iOS SDK里就有众多的delegate,比如最常用的UITableView,
我们没法知道Apple怎么重用UITableViewCell,怎么处理UITableView里Cell的增加、删减,因为我们没有源码。
但是我们可以通过实现Delegate的方法来控制一个UITableView的一些行为。
UITableViewDataSource其实和delegate是一样一样的,只是由于意义不同换了个名字罢了。
protocol在此扮演了什么角色呢?
protocol是一种语法,它提供了一个很方便的、实现delegate模式的机会。
比如写UITableView的时候,Apple这么干
UITableView.m
- (void)doSomething {[self blahblah];[self.delegate guruguru];[self blahblah];}
delegate是我们写的类,这个类如果可以被传给UITableView做为其delegate,那唯一要求,就是它实现了
- (void)guruguru;
这个方法。
如果我们把这个方法定义在一个protocol里
@protocol XXXProtocol- (void)guruguru;@end
就说明了,UITableView需要的delegate是一个conform to XXXProtocol的类。
这就正好是
id<XXXProtocol>
表达的意思。
无论具体的类是什么,它还有其它什么方法,只要它conform to这个protocol,
就说明它可以被传给UITableView,作为它的delegate。
那么Apple为了让我们知道这个protocol是delegate需要conform的protocol,
它就把XXXProtocol改成了UITableViewDelegate
这样我们看到protocol的名字里有Delegate,就知道这个protocol里的函数是用来做自定义(Customization)的了。
来源:http://haoxiang.org/2011/08/ios-delegate-and-protocol/
Protocol 的其它问题
1. 使用时为什么要加上 iOS.delegate = self
物件名称.delegate = self,是在採用任何协定时 一定会看到的一行程式码,由于定义协定的类别并不需要实作协定内的方法,因为实作的部份是由採纳协定的类别来实作,但是它又必须要知道是由哪一个类别来实作,因此我们必须要把採纳协定类别的 instance 交给定义协定的类别,让它来使用。
另一方面并不是任何类别都可以将 instance 传给定义协定的类别来使用,其原因是,我们在定义此协定的类别里有宣告 delegate 变数时,有限定它必须要採纳此协定(id delegate)如果没有採用该协定就将 instance 传给定义该协定的类别,Xcode 同样会发出警告讯息。
2. 为什么协定的生效位置不能写在建构式中
协定的生效位置写在建构式中,并不会造成程式编译上的任何问题,因为这是属于逻辑上的错误,协定要正常生效它必须要知道实作它方法的类别的 instance,如果将生效的位置写在建构式中,在建立定义此协定的形态的变物件时,它的确会去触发此协定内的方法,但是由于并没有给它实作此协定方法类别的 instance,因此不会有任何效果产生,反之,如果一定要将生效的位置写在建构式中,那么在初始化时就必须要设定好 delegate 才行,也就是使用初始化的方法函式里还必须要带入一个参数物件好指定给 delegate。
3. 在定义协定时同时也可以採用其他的协定
如果在定义协定时同时又採用其他的协定,这会导致之后採纳此协定的类别,它必须同时实作出两个协定内的方法,同样地,你也可以利用此方式来扩充那些已经存在的协定。
@protocol FurnaceDelegate <其它可能的协定名称>
4. 使用 @optional 提供选择性的实作
@optional,如同它字面上的意义,在 @optional 之后的方法都可以是选择性的实作,在定义协定时使用此方法,可以让之后採纳此协定的类别不一定要完全实作出协定内的所有方法。
@protocol FurnaceDelegate - (void)whenCalledDelegeteFunction;@optional-(void)optionalDelegeteFunction;@end
来源:http://furnacedigital.blogspot.com/2012/01/protocol.html
delegate和protocol相关推荐
- category、protocol、delegate总结
一.category(类别) 类别是一种类扩展的机制,能为现有的类添加新方法. ①类别的基本语法 @interface部分,如: @interface NSString (NumberConvenie ...
- 传参方法:sharedApplication, NSUserDefaults, protocol 和 delegate(实例)
1. iOS开发中使用[[UIApplication sharedApplication] openURL:] 加载其它应用 在iOS开发中,经常需要调用其它App,如拨打电话.发送邮件等.UIApp ...
- segue和delegate实现两个页面传值
页面传值 如果有两个页面A和B之前需要传值,如下图所示,从A页面打开一个B页面,填入一些信息后,展现在B页面.这个过程需要从B页面传数据给A页面. A->B的数据传递可以在segue中实现,B- ...
- 如何理解 Objective-C Delegate
例如,我们要在一个 ViewController 中使用一个ActionSheet,代码如下: UIActionSheet *actionSheet = [[UIActionSheet alloc]i ...
- Notification与Delegate实现通讯沙拉实例
这个例子是我在AppStore上下的一个沙拉的app模仿做的,现在还没做文件管理. 黄星星就是收藏的意思,在navigationBar上面有个按钮查看收藏夹内容的,黄星星是做了个button,开始想做 ...
- iOS传值之代理传值
iOS中传值方式有好几种,分别是:代理传值,block传值,属性传值,通知传值,单例传值,利用userdefault或者文件方式传值,通常代理传值和block传值使用最普遍,本文介绍代理传值的方式,后 ...
- (IOS)截图Demo
思路是建一个UIView的子类,获取划动出的矩形,用协议将矩形传递给代理对象,依据该矩形完成图像数据的截取,并显示出来. 截图视图类: #import <UIKit/UIKit.h>@pr ...
- iOS VIPER架构(三)
路由是实现模块间解耦的一个有效工具.如果要进行组件化开发,路由是必不可少的一部分.目前iOS上绝大部分的路由工具都是基于URL匹配的,优缺点都很明显.这篇文章里将会给出一个更加原生和安全的设计,这个设 ...
- 0113——代理模式
什么是代理? 苹果的官方文档给了很清晰的解释: Delegation is a simple and powerful pattern in which one object in a program ...
最新文章
- 谷歌发布颠覆性研究:不训练不调参,AI自动构建超强网络,告别炼丹一大步...
- 百度要在“元宇宙”里办AI开发者大会了
- opengl 粒子按轨迹运动_袁讲经典4:一个粒子在电场中的运动轨迹相关问题
- 大牛深入浅出讲解C语言#define宏定义应用及使用方法
- 2个网页跳来跳去_怎么写最优化的网页标题标签(Title Tag)?
- wxWidgets:你好,世界
- Go基础编程:环境搭建
- 转:华为副总裁一封信:透露年薪千万的工作感悟,太震撼了!
- 夜视模式,多少猥琐相机假汝之名
- C++获取本机的ip地址程序
- android梅花形布局,Android相对布局实现各种梅花效果
- kali201904更新国内源的步骤
- select框多级联动
- 论文参考文献正确插入方法 (一)
- Spring学习02-Spring中的设计模式(一)
- 鹿晗关晓彤公开恋情,是如何把微博服务器搞炸的? 鹿晗关晓彤公开恋情,是如何把微博服务器搞炸的? 题图:《盗墓笔记》 鹿晗关晓彤公开恋情,是如何把新浪微博的服务器搞垮的? 知友:苏莉安(200+
- 适配iOS 11和iPhone X——导航栏、UITableView
- android媲美微信扫码库
- linux系统下怎么安装.deb文件?
- 室内定位UWB在化工园区如何智能化管理
热门文章
- js取iframe 上级页面_原生js获取iframe中dom元素--父子页面相互获取对方dom元素的方法...
- MobileNet V3简单总结
- [Unity] GameFramework 学习记录 3
- ospf避免环路_路由环路知识点总结!
- 酷黑风个人主页+引导页
- python做什么方向好_Python工程师的择业方向有哪些?你想好做什么工作了吗?
- python数组索引和切片_python numpy数组的索引和切片的操作方法
- 七彩影视双端新版本源码
- 爱链工具 v1.11.24.0
- 米酷影视直播二开全修复版增加在线充值影视资源网站源码