Objective-C 学习笔记 - part 4 - 协议

**以下内容中 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

posted on 2011-08-14 23:09  沙加  阅读(849)  评论(0编辑  收藏  举报

导航