**以下内容中 interface 指代传统意义上的接口, 而@interface 指代 Objective-C 中的类型规范概念,以免混乱。
看到这一章,有点奇怪,其实只要将 protocal 当作 C# 里面的接口来认识就好了。
因为 Objective-C 给每个 Class 都分配了一个 @interface ,因此用这人概念来表达 interface 的意思。
抽象出接口的目的:
1. 相似方法组的实现有共同的规范。 2. 让“接口”独立于类。 3. 封装没有继承关系的类的共性。
if ( [assistant respondsToSelector:@selector(helpOut:)] ) { [assistant helpOut:self]; return YES; } return NO;
上面的代码是不是可以认为使用”反射“确定了一个对象是否支持一个方法调用。
但是 protocal 比 interface 强大的地方在于,它是可以跨应用程序的,这也是之所以叫它 protocal 的原因。
事实上,将“方法调用”都转换为消息传递本身就是弱化了应用程序域的概念,让多个程序可以无缝地工作在一起,在这一点上苹果确实是很高明的。
匿名类也可以利用 protocal 来与外界沟通而无需定义与之对应的 @interface。
如何申明一个 protocal: @protocol ProtocolName method declarations @end
需要注意,协议不象类名称一样全局可见,它只存于作用的命名空间中。
协议用的方法可以申明为 option 方式(可选/非强制方式)
非正式的协议使用 @interface 来申明,如: @interface NSObject ( MyXMLSupport ) - initFromXMLRepresentation:(NSXMLElement *)XMLElement; - (NSXMLElement *)XMLRepresentation; @end
与类对应的 @interface 不一样的是,非正式协议没有对应的实现文件,即只有一个空的协议在那里,可以认为它是用来标定一类有相似行为的类的分组方法。
非正式协议没有编译时的类型检查,如果需要类型检查,应该使用正式协议。
非正式协议常用于所有方法都是可选时的情形,但对于 OS X > 10.5 最好使用正式协议。
关于协议对象:
正式协议是一种特殊的数据类型:它闪是 Protocal 的实例而非一般对象是 Class 的实例。
运行时只初化有引用的 @protocal
关于采用一个协议;
不论正式或非正式协议都使用下面的语法来申明: @interface ClassName : ItsSuperclass < protocol list >
使用多个协议的情况: @interface Formatter : NSObject < Formatting, Prettifying >
使用协议的类需要 include 协议所在的 .h 文件,并且协议中的方法不义定义在 类型自己的@interface 中,这样来避免重复。
一个类可以没有对应的 @interface 而只有 protocal
可以使用下面的测试方法来检查一个对象是否遵守某个 protocal: if ( ! [receiver conformsToProtocol:@protocol(MyXMLSupport)] )
如果只需要检查一个特定方法的话,尽量使用 respondsToSelector: test 从则避免检测 protocal 中的所有方法。
协议可以继承
由于协议可以互相引用, 因此在某些情况下不需要 import 协议所在的文件,而只需要象这样: @protocol B; @protocol A - foo:(id <B>)anObject; @end
|
请发表评论