Objective-C 复合对象

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


类集群

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

为了简单起见,我们不为类似函数创建多个类,而是创建一个基于输入值处理的类。

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


什么是复合对象?

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

下图对此进行了说明。

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

NSArray 类的 count 方法就是一个例子;介入对象对其重写的方法的实现可以像下面这样简单:

  1. - (unsigned)count {
  2. return [embeddedObject count];
  3. }

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


一个复合对象实例

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

  1. #import <Foundation/Foundation.h>
  2. @interface ValidatingArray : NSMutableArray {
  3. NSMutableArray *embeddedArray;
  4. }
  5. + validatingArray;
  6. - init;
  7. - (unsigned)count;
  8. - objectAtIndex:(unsigned)index;
  9. - (void)addObject:object;
  10. - (void)replaceObjectAtIndex:(unsigned)index withObject:object;
  11. - (void)removeLastObject;
  12. - (void)insertObject:object atIndex:(unsigned)index;
  13. - (void)removeObjectAtIndex:(unsigned)index;
  14. @end
  15. @implementation ValidatingArray
  16. - init {
  17. self = [super init];
  18. if (self) {
  19. embeddedArray = [[NSMutableArray allocWithZone:[self zone]] init];
  20. }
  21. return self;
  22. }
  23. + validatingArray {
  24. return [[self alloc] init] ;
  25. }
  26. - (unsigned)count {
  27. return [embeddedArray count];
  28. }
  29. - objectAtIndex:(unsigned)index {
  30. return [embeddedArray objectAtIndex:index];
  31. }
  32. - (void)addObject:(id)object {
  33. if (object != nil) {
  34. [embeddedArray addObject:object];
  35. }
  36. }
  37. - (void)replaceObjectAtIndex:(unsigned)index withObject:(id)object; {
  38. if (index <[embeddedArray count] && object != nil) {
  39. [embeddedArray replaceObjectAtIndex:index withObject:object];
  40. }
  41. }
  42. - (void)removeLastObject; {
  43. if ([embeddedArray count] > 0) {
  44. [embeddedArray removeLastObject];
  45. }
  46. }
  47. - (void)insertObject:(id)object atIndex:(unsigned)index; {
  48. if (object != nil) {
  49. [embeddedArray insertObject:object atIndex:index];
  50. }
  51. }
  52. - (void)removeObjectAtIndex:(unsigned)index; {
  53. if (index <[embeddedArray count]) {
  54. [embeddedArray removeObjectAtIndex:index];
  55. }
  56. }
  57. @end
  58. int main() {
  59. NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
  60. ValidatingArray *validatingArray = [ValidatingArray validatingArray];
  61. [validatingArray addObject:@"Object1"];
  62. [validatingArray addObject:@"Object2"];
  63. [validatingArray addObject:[NSNull null]];
  64. [validatingArray removeObjectAtIndex:2];
  65. NSString *aString = [validatingArray objectAtIndex:1];
  66. NSLog(@"The value at Index 1 is %@",aString);
  67. [pool drain];
  68. return 0;
  69. }

结果如下:

  1. 2022-07-07 22:03:54.294 demo[6247] The value at Index 1 is Object2

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