禅与 Objective-C 编程艺术学习笔记<一> -- 类簇

《禅与 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

这个实例展示了如何创建一个类簇。

  1. isMemberOfClass:方法的作用是判断是否是这个类的实例,当[[ZOCKintsugiPhotoViewController alloc] initWithPhotos:photos] 被调用时,上面条件表达式的结果将会是 true,而使用它的子类,上面表达式结果将返回 false。这样 使用 [self isMemberOfClass:ZOCKintsugiPhotoViewController.class]可以 防止子类中重载初始化方法,避免无限递归。

  2. self = nil 的目的是移除 ZOCKintsugiPhotoViewController 实例上的所有引用,实例(抽象类的实例)本身将会解除分配( 当然ARC也好MRC也好dealloc都会发生在Main Runloop这一次的结束时)。

  3. 接下来的逻辑就是判断哪一个私有子类将被初始化。

最后,不管是在iPhone上还是在iPad上,我只需使用类似下面的代码,而不用管具体用的是哪个子类。

    NSArray *photos = @[];
    ZOCKintsugiPhotoViewController *photoViewVC = [[ZOCKintsugiPhotoViewController alloc] initWithPhotos:photos];
    NSLog(@"photoViewVC:%@",photoViewVC);

例如,我在iPhone下运行,输出的类名是 ZOCKintsugiPhotoViewController_iPhone

ClassName.png

另外,还有几篇篇关于类簇写得不错的博文,可以帮助我们更深层次的加强理解,有兴趣的可以去看下:

  • sunnyxx 的 从NSArray看类簇
  • Limboy的 类簇在iOS开发中的应用
  • 一个iOS菜菜的白话文记录 - Class Cluster 类簇

你可能感兴趣的:(禅与 Objective-C 编程艺术学习笔记<一> -- 类簇)