第9条:以“类族模式”隐藏实现细节
1. 类族模式
“类族”(class cluster)是一种模式(pattern)。该模式可以灵活应对多个类,将它们的实现细节隐藏在“抽象基类”(abstract base class)后面,以保持接口简洁。开发者无须自己创建子类实例,只需调用基类方法来创建即可。
2. 创建类族
EOCEmployee头文件:
/* 抽象基类 */
// 职员的类型
typedef NS_ENUM(NSUInteger,EOCEmployeeType) {
EOCEmployeeTypeDeveloper,
EOCEmployeeTypeDesigner,
EOCEmployeeTypeFinance,
};
@interface EOCEmployee : NSObject
/** 名字 */
@property(nonatomic,copy) NSString *name;
/** 薪水 */
@property(nonatomic,assign) NSUInteger salary;
// 工厂方法(类方法):创建雇员对象
+ (EOCEmployee *)emplyeeWithType:(EOCEmployeeType)type;
// 职员干自己的工作
- (void)doADaysWork;
@end
EOCEmployee实现文件:
#import "EOCEmployee.h"
#import "EOCEmployeeDeveloper.h"
#import "EOCEmployeeDesigner.h"
#import "EOCEmployeeFinance.h"
@implementation EOCEmployee
/* 将子类的实例的创建隐藏在基类的实现方法中 */
// 根据职员类型创建子类各自的实例
+ (EOCEmployee *)emplyeeWithType:(EOCEmployeeType)type{
switch (type) {
case EOCEmployeeTypeDeveloper:
return [EOCEmployeeDeveloper new];
break;
case EOCEmployeeTypeDesigner:
return [EOCEmployeeDesigner new];
break;
case EOCEmployeeTypeFinance:
return [EOCEmployeeFinance new];
break;
}
}
- (void)doADaysWork{
// 在子类的实现文件中各自实现其工作
}
@end
EOCEmployee的子类EOCEmployeeDeveloper的实现文件:
#import "EOCEmployeeDeveloper.h"
@implementation EOCEmployeeDeveloper
- (void)doADaysWork{
// 子类其工作的实现细节
[self writeCode];
}
- (void)writeCode{
NSLog(@"writeCode");
}
@end
main函数:
#import "EOCEmployee.h"
int main(int argc, const char * argv[]) {
@autoreleasepool {
EOCEmployee *developer = [EOCEmployee emplyeeWithType:EOCEmployeeTypeDeveloper];
NSLog(@"%@",[developer class]);// output EOCEmployeeDeveloper
/*
* 总结:
* 工厂模式
* 通过EOCEmployee类的工厂方法创建出来的实例是EOCEmployee类的子类的实例
*/
}
return 0;
}
3. Cocoa里的类族
系统框架中有许多类族,大部分collection类都是某个类族中的抽象基类。NSArray与NSMutableArray实际上有两个抽象基类,但是仍然算是一个类族,意味着两者在实现各自类型的数组时可以共用实现代码,此外,还能把可变数组复制为不可变数组,反之亦然。
id maybeAnArray = /* ... */;
if ([maybeAnArray class] == [NSArray class]){// 永远不可为真
// will nerver be hit
}
解释:[maybeAnArray class] 所返回的类绝不可能是NSArray类本身,因为由NSArray的初始化方法所返回的那个实例所属的类型是“隐藏在类族公共接口后面的那个内部类型”。
要判断出某个实例所属的类是否位于类族之中需要用类型信息查询方法(自省)。
如:
id maybeAnArray = /* ... */;
if (maybeAnArray isKindOfClass:[NSArray class]){
// will be hit
}
*** 向Cocoa中NSArray这样的类族新增子类所要遵守的规则 ***
- 子类应该继承自类族中的抽象基类。
- 子类应该定义自己的数据存储方式。
子类必须用一个实例变量来存放数组中的对象,而NSArray本身只不过是包在其他隐藏对象外面的壳,它仅仅定义了所有数组都需具备的一些接口。对于这个自定义的数组子类来说,可以用NSArray来保存其实例。 - 子类应当覆写超类文档中指明需要覆写的方法。
在类族中实现子类时所需遵循的规范一般都会定义基类的文档之中,编码前应该先看看。
要点
- 类族模式可以把实现细节隐藏在一套简单的公共接口(抽象基类)后面。
- 系统框架中经常使用类族。
- 从类族的公共抽象基类中继承子类时要当心,若有开发文档,则应首先阅读。