《禅与 Objective-C 编程艺术学习》是一本开源的电子书,你可以把它当做一本编程规范的书来看,也可以看做是一本 Effective Objective-C,电子书传送门:禅与 Objective-C 编程艺术学习,非常推荐大家看一看。
类簇介绍(Class Clusters)
类簇在Apple的文档中这样描述:
an architecture that groups a number of private, concrete subclasses under a public, abstract superclass. (一个在共有的抽象超类下设置一组私有子类的架构)
如果这个描述听起来很熟悉,说明你的直觉是对的。 Class cluster 是 Apple 对抽象工厂设计模式的称呼。
Class Clusters 的想法很简单: 使用信息进行(类的)初始化处理期间,会使用一个抽象类(通常作为初始化方法的参数或者判定环境的可用性参数)来完成特定的逻辑或者实例化一个具体的子类。而这个"Public Facing(面向公众的)"类,必须非常清楚他的私有子类,以便在面对具体任务的时候有能力返回一个恰当的私有子类实例。对调用者来说只需知道对象的各种API的作用即可。这个模式隐藏了他背后复杂的初始化逻辑,调用者也不需要关心背后的实现。
Class Clusters 在 Apple 的Framework 中广泛使用:一些明显的例子比如 NSNumber
可以返回不同类型给你的子类,取决于 数字类型如何提供 (Integer, Float, etc...) 或者 NSArray
返回不同的最优存储策略的子类。
这个模式的精妙的地方在于,调用者可以完全不管子类,只需使用简单地接口,就可以得到实际的返回的类,而不用去管相关的细节。
我们的经验是使用类簇可以帮助移除很多条件语句。
类簇实例
一个经典的例子是如果你有为 iPad 和 iPhone 写的一样的 UIViewController
子类,但是在不同的设备上有不同的行为。
比较基础的实现是用条件语句检查设备,然后执行不同的逻辑。虽然刚开始可能不错,但是随着代码的增长,运行逻辑也会趋于复杂。 一个更好的实现的设计是创建一个抽象而且宽泛的 view controller 来包含所有的共享逻辑,并且对于不同设备有两个特别的子例。
通用的 view controller 会检查当前设备并且返回适当的子类。
@implementation ZOCKintsugiPhotoViewController
- (id)initWithPhotos:(NSArray *)photos
{
// 1.
if ([self isMemberOfClass:ZOCKintsugiPhotoViewController.class]) {
//2.
self = nil;
//3.
if ([UIDevice isPad]) {
self = [[ZOCKintsugiPhotoViewController_iPad alloc] initWithPhotos:photos];
}
else {
self = [[ZOCKintsugiPhotoViewController_iPhone alloc] initWithPhotos:photos];
}
return self;
}
return [super initWithNibName:nil bundle:nil];
}
@end
@implementation ZOCKintsugiPhotoViewController_iPad
- (instancetype)initWithPhotos:(NSArray *)photos {
//...
}
@end
@implementation ZOCKintsugiPhotoViewController_iPhone
- (instancetype)initWithPhotos:(NSArray *)photos {
//...
}
@end
这个实例展示了如何创建一个类簇。
isMemberOfClass:
方法的作用是判断是否是这个类的实例,当[[ZOCKintsugiPhotoViewController alloc] initWithPhotos:photos]
被调用时,上面条件表达式的结果将会是true
,而使用它的子类,上面表达式结果将返回false
。这样 使用[self isMemberOfClass:ZOCKintsugiPhotoViewController.class]
可以 防止子类中重载初始化方法,避免无限递归。self = nil
的目的是移除ZOCKintsugiPhotoViewController
实例上的所有引用,实例(抽象类的实例)本身将会解除分配( 当然ARC也好MRC也好dealloc都会发生在Main Runloop这一次的结束时)。接下来的逻辑就是判断哪一个私有子类将被初始化。
最后,不管是在iPhone上还是在iPad上,我只需使用类似下面的代码,而不用管具体用的是哪个子类。
NSArray *photos = @[];
ZOCKintsugiPhotoViewController *photoViewVC = [[ZOCKintsugiPhotoViewController alloc] initWithPhotos:photos];
NSLog(@"photoViewVC:%@",photoViewVC);
例如,我在iPhone下运行,输出的类名是 ZOCKintsugiPhotoViewController_iPhone
另外,还有几篇篇关于类簇写得不错的博文,可以帮助我们更深层次的加强理解,有兴趣的可以去看下:
- sunnyxx 的 从NSArray看类簇
- Limboy的 类簇在iOS开发中的应用
- 一个iOS菜菜的白话文记录 - Class Cluster 类簇