Objective-C 复合对象

  • 复合对象

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

    类集群是基础框架广泛使用的一种设计模式。类集群将一些私有的具体子类归为一个公共抽象超类。以这种方式对类进行分组可简化面向对象框架的公共可见架构,而不会降低其功能丰富性。类群集基于抽象的工厂设计模式。为简单起见,我们没有创建类似功能的多个类,而是创建了一个单个类,该类将根据输入的值来处理它。例如,在NSNumber中,我们有许多类簇,例如char,int,bool等。我们将所有这些分组到一个类中,该类负责在一个类中处理类似的操作。NSNumber实际上将这些原始类型的值包装到对象中。
  • 什么是复合对象

    通过将私有群集对象嵌入我们自己设计的对象中,我们创建了一个复合对象。该复合对象可以依靠集群对象的基本功能,仅拦截复合对象想要以某种特定方式处理的消息。这种体系结构减少了我们必须编写的代码量,并使您可以利用Foundation Framework提供的经过测试的代码。下图对此进行了说明。
    复合对象必须声明自己是集群抽象超类的子类。作为子类,它必须重写超类的原始方法。它也可以覆盖派生的方法,但这不是必需的,因为派生的方法通过原始方法工作。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;
    }
    
    现在,当我们编译并运行程序时,我们将得到以下结果。
    
    2020-08-24 11:07:14.546 helloWorld[7324:14668] The value at Index 1 is Object2
    
    在上面的示例中,我们可以看到验证数组的one函数将不允许添加空对象,这将导致正常情况下崩溃。但是我们的验证数组会处理它。类似地,验证数组中的每个方法除了正常的操作顺序外,还会添加验证过程。