组合模式(Composite Pattern)

组合模式:允许你将对象组合成树形结构来表现“整体/部分”层次的结构。组合能让客户以一致的方式处理个别对象以及对象组合。

组合模式是结构型模式之一。组合模式为所有的对象定义相同的接口,客户对单个对象和组合对象具有一致性。

将原本节点之间复杂的逻辑关系放在组合之中,客户无需关系组合节点之间的关系。这是以牺牲单一职责原则换来的。

我们不难发现,组合模式不仅要处理对象原本的功能,还要管理节点之间的逻辑关系。

iOS/Android中我们的页面逻辑,视图(view)就是采用的组合模式
。window是根节点,再添加Controller/Activity中的view。然后再添加视图内的子视图,一层套一层,这就是一个树形结构。而我们处理的时候只需要处理当前的视图逻辑即可。比如我们点击屏幕,会一层层的传递点击事件,也是以组合获取子视图,然后一层层的传递出去的,直至有视图响应事件。

组合模式

举个栗子
假设一个国家要划分行政区。大的行政区下又会又小的行政区。
我们这里不按国内的来这个来,我们以国家,省(包括直辖市),市3级来划分,那么这就是一个组合的栗子。
我们想知道这个国家的总的行政划分(直接调用description)
国家是树根节点,省(包含直辖市)是分支节点,市(直辖市的区)是叶子节点。

首先我们创建一个区域的协议(java/Android中是接口)

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@protocol Area <NSObject>

@property (nonatomic ,strong)NSString * name;

-(void)add:(id<Area>)area;

-(void)remove:(id<Area>)area;

-(id<Area>)getChild:(NSInteger)index;

@end

NS_ASSUME_NONNULL_END

这是 国家和省(包含直辖市)的逻辑,这里整体逻辑比较简单,所以国家和省的逻辑是相同的。

#import "Country.h"

@interface Country ()

@property (nonatomic ,strong)NSMutableArray <Area>* areaArray;
@end

@implementation Country

@synthesize name;

-(instancetype)init {
    
    if (self = [super init]) {
        
        _areaArray = (NSMutableArray <Area>*)[NSMutableArray array];
    }return self;
}

-(void)add:(nonnull id<Area>)area {
    
    [_areaArray addObject:area];
}

- (id<Area>)getChild:(NSInteger)index {
    
    if (index >=0 && index <= self.areaArray.count - 1) {
        return _areaArray[index];
    }
    return nil;
}


- (void)remove:(nonnull id<Area>)area {
    
    if ([_areaArray containsObject:area]) {
        [_areaArray removeObject:area];
    }
}

-(NSString *)description {
    
    NSMutableArray * desArray = [NSMutableArray array];
    for (id<Area> area in _areaArray) {
        [desArray addObject:area.description];
    }
    return [NSString stringWithFormat:@"<%@:%p,name=%@,%@>",[self class],self,name,desArray];
    
}

@end

市是最小一级的行政区划,所以没有子节点

#import "County.h"


@implementation County

@synthesize name;


-(void)add:(nonnull id<Area>)area {
    
    @throw [NSException exceptionWithName:@"不能添加行政区域" reason:@"市是最小的一级行政规划" userInfo:nil];
}

- (id<Area>)getChild:(NSInteger)index {
    
    @throw [NSException exceptionWithName:@"没有更小行政区域" reason:@"市是最小的一级行政规划" userInfo:nil];
}


- (void)remove:(nonnull id<Area>)area {
    
    @throw [NSException exceptionWithName:@"没有更小行政区域" reason:@"市是最小的一级行政规划" userInfo:nil];
}

-(NSString *)description {
    
    return [NSString stringWithFormat:@"<%@:%p,name=%@>",[self class],self,name];
}

@end

具体调用,并打印行政规划

#import <Foundation/Foundation.h>
#import "Area.h"
#import "Country.h"
#import "County.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
        id<Area> country = [[Country alloc] init];
        country.name = @"中国";
        
        id<Area> province = [[Country alloc] init];
        province.name = @"河北省";
        
        id<Area> province1 = [[Country alloc] init];
        province.name = @"河南省";
        
        [country add:province];
        
        [country add:province1];
        
        id<Area> county = [[County alloc] init];
        county.name = @"张家口市";
        
        id<Area> county1 = [[County alloc] init];
        county.name = @"承德市";
        [province add:county];
        [province add:county1];
        
        id<Area> county3 = [[County alloc] init];
        county.name = @"郑州市";
        
        id<Area> county4 = [[County alloc] init];
        county.name = @"开封市";
        [province1 add:county3];
        [province1 add:county4];
        
        NSLog(@"%@",country);
    }
    return 0;
}

优点

高层模块调用简单
节点自由增加

缺点

层次关系维护导致设计复杂
违背了依赖倒置原则,和单一职责原则

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容