Object-C Categories和Protocols

Categoryjava

要扩展一个不可修改的类,一般的作法是为该类建立一个子类,在子类中实现想要实现的方法,在Object-C中,能够经过category来实现,而且实现方式更为简单。框架

如今有以下定义:一个包含4个方法的Fraction类ui

1 @interface Fraction : NSObject
2 @property int num, den;
3 
4 -(void) setTo:(int) n over:(int) d;
5 -(void) reduce;
6 -(double) converToNum;
7 -(void) print;
8 @end

如今要为该类增长4个方法,分别为加法、减法、乘法和除法:spa

 1 // 定义category文件:FractionMathOps.h文件
 2 // 因为category是扩展文件,全部首先须要引入原文件
 3 #import "Fraction.h"
 4 
 5 @interface Fraction (MathOps)
 6 -(Fraction *) add: (Fraction *) f;
 7 -(Fraction *) mul: (Fraction *) f;
 8 -(Fraction *) sub: (Fraction *) f;
 9 -(Fraction *) div: (Fraction *) f;
10 @end
11 
12 // 可选的将implementation定义在interface文件中,或者定义在单独的FractionMathOps.m文件中
13 @implementation Fraction (MathOps)
14 -(Fraction *) add : (Fraction *) f
15 {
16     ....
17 }
18 ....
19 @end

经过上述方法定义的category不能为原类追加成员变量,若须要追加成员变量,在定义category时不能填写名字,如:code

 1 #import "Fraction.h"
 2 
 3 @interface Fraction ()
 4 @property int uniqueID;
 5 
 6 -(void) doStuffWithUniqueID: (int) theID;
 7 @end
 8 
 9 // -------------------------------------------------------
10 
11 // implementation实现部分,注意,Fraction后面没有括号()
12 @implementation Fraction
13 @synthesize uniqueID;
14 
15 -(void) doStuffWithUniqueID: (int) theID
16 {
17     ...
18 }
19 @end

经过这种方法为类追加的全部方法都是private属性的,因此只但愿类本事使用这些方法,这是一个好办法。orm

若使用category追加原类中已经方法时,原类中的方法会被覆盖,且没法调用原类中的方法。对象

能够为一个类建立多个category,若多个category文件中定义了同一个方法,Object-C没法肯定具体会调用那个方法。blog

同时须要注意,为一个类添加category,受影响的不单单是该类,也包括该类的子类在内。继承

Protocols和Delegation接口

Protocol是指一系列可在多个类中共享的方法,这些方法并无特定的implementation去实现,其实就是java中的接口。

protocol中的方法有些能够选择性的实现,有些则必须被实现

定义protocol,只须要使用@protocol,后面接protocol的名字,以后开始像在interface中声明方法那样声明protocol方法,全部方法声明之后,以@end结尾。

在Foundation框架中,

1 @protocol NSCopying
2 - (void)copyWithZone: (NSZone *)zone;
3 @end 

定义一个是否须要必须被实现的方法:@required和@optional,默认为@required。

@protocol MyProtocol
@required // 该命令能够省略
- (void) print1; // 必须被实现
@optional 
- (void) print2; // 能够不被实现
@end

一个类若要实现一个或多个protocol,只需将protocolName放到尖括号"<>"中追加到@interface后面既可,多个protocols之间使用逗号","分割。

1 @interface MyObj: NSObject <NSCopying>

 在上述代码中,因为系统已经知道了NSCopying的定义,因此无需在interface中定义NSCopying protocol。

若一个类实现了一个或多个protocol,该类的子类一样会实现这些protocols

可使用conformsToProtocol:方法去判断一个对象是否实现了指定的protocol,如:

1 id currentObj;
2 ...
3 if ([currentObj conformsToProtocol: @protocol (Drawing)] == YES)
4 {
5     ...
6 }

在上述代码中,经过使用@protocol (protocolName)的方式产生一个protocol对象 

1 // 声明一个必须实现Drawing protocol的类实例
2 // 若currentObj类中没有实现给protocol,则编译器会发出警告信息
3 id <Drawing> currentObj;

定义一个protocol时,能够指定该protocol继承另外一个protocol:

1 // MyPro1 继承MyPro
2 // 此时实现MyPro1的类也必须实现MyPro中必需要被实现的方法
3 @protocol MyPro1 <MyPro>
4 ...
5 @end

在category中实现protocol

1 @interface MyObj (stuff) <MyPro>
2 ...
3 @end

能够将MyPro的实现放到stuff中

1 Fraction *frac=[[Fraction alloc] initWithNumerator: 3 denominator: 5];
2 <Printing1> p1=frac;
3 [p1 print1];
4 id<Printing1,Printing2,Printing3> p2=frac;
5 [p2 print2];
6 [p2 print3];
相关文章
相关标签/搜索