Objective-C 复合对象

我们可以在类集群中创建子类,该类集群定义了一个在其中嵌入对象的类。 这些类对象是复合对象。 所以你可能想知道什么是类集群。 因此,我们将首先了解什么是类集群。


类集群

类集群是基础框架广泛使用的一种设计模式。 类簇将许多私有具体子类分组在一个公共抽象超类下。 以这种方式对类进行分组简化了面向对象框架的公开可见体系结构,而不会降低其功能丰富性。 类集群基于抽象工厂设计模式。

为简单起见,我们没有为相似的功能创建多个类,而是创建了一个类来根据输入的值进行处理。

例如,在 NSNumber 中,我们有许多类的集群,如 char、int、bool 等。 我们将所有这些分组到一个类中,该类负责处理单个类中的类似操作。 NSNumber 实际上是将这些原始类型的值包装成对象。


什么是复合对象?

通过在我们自己设计的对象中嵌入私有集群对象,我们创建了一个复合对象。这个复合对象可以依赖于集群对象的基本功能,只拦截复合对象想要以某种特定方式处理的消息。 这种架构减少了我们必须编写的代码量,并让您可以利用 基础架构 提供的经过测试的代码。

下图解释了这一点。

Objective-C 复合对象

复合对象必须声明自己是集群抽象超类的子类。 作为子类,它必须覆盖超类的原始方法。 它还可以覆盖派生方法,但这不是必需的,因为派生方法通过原始方法工作。

以NSArray类的count方法为例; 干预对象对它重写的方法的实现可以很简单 −

- (unsigned)count  {
   return [embeddedObject count];
}

在上面的例子中,嵌入对象实际上是NSArray类型。


复合对象示例

现在为了查看完整的示例,让我们看一下下面给出的 Apple 文档中的示例。

#import <Foundation/Foundation.h>

@interface ValidatingArray : NSMutableArray {
   NSMutableArray *embeddedArray;
}

+ validatingArray;
- init;
- (unsigned)count;
- objectAtIndex:(unsigned)index;
- (void)addObject:object;
- (void)replaceObjectAtIndex:(unsigned)index withObject:object;
- (void)removeLastObject;
- (void)insertObject:object atIndex:(unsigned)index;
- (void)removeObjectAtIndex:(unsigned)index;

@end

@implementation ValidatingArray
- init {
   self = [super init];
   if (self) {
      embeddedArray = [[NSMutableArray allocWithZone:[self zone]] init];
   }
   return self;
}

+ validatingArray {
   return [[self alloc] init] ;
}

- (unsigned)count {
   return [embeddedArray count];
}

- objectAtIndex:(unsigned)index {
   return [embeddedArray objectAtIndex:index];
}

- (void)addObject:(id)object {
   if (object != nil) {
      [embeddedArray addObject:object];
   }
}

- (void)replaceObjectAtIndex:(unsigned)index withObject:(id)object; {
   if (index <[embeddedArray count] && object != nil) {
      [embeddedArray replaceObjectAtIndex:index withObject:object];
   }
}

- (void)removeLastObject; {
   if ([embeddedArray count] > 0) {
      [embeddedArray removeLastObject];
   }
}

- (void)insertObject:(id)object atIndex:(unsigned)index; {
   if (object != nil) {
      [embeddedArray insertObject:object atIndex:index];
   }
}

- (void)removeObjectAtIndex:(unsigned)index; {
   if (index <[embeddedArray count]) {
      [embeddedArray removeObjectAtIndex:index];
   }
}

@end

int main() {
   NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
   ValidatingArray *validatingArray = [ValidatingArray validatingArray];
   
   [validatingArray addObject:@"Object1"];
   [validatingArray addObject:@"Object2"];
   [validatingArray addObject:[NSNull null]];
   [validatingArray removeObjectAtIndex:2];
   NSString *aString = [validatingArray objectAtIndex:1];
   NSLog(@"The value at Index 1 is %@",aString);
   [pool drain];
   
   return 0;
}

现在当我们编译并运行程序时,我们将得到如下结果。

2013-09-28 22:03:54.294 demo[6247] The value at Index 1 is Object2

在上面的示例中,我们可以看到验证数组的一个函数不允许添加 null 对象,这将导致在正常情况下崩溃。但是我们的验证数组会处理它。 类似地,验证数组中的每个方法都在正常操作序列之外添加了验证过程。