设计模式系列7--组合模式

场景分析

我们平时去餐厅吃饭,都会使用菜单来点餐,今天我们来实现一个超级菜单,这个一个菜单大集合,包括单一菜品和子菜单,如图所示:

image

可以看到上面的菜单不但包括单个的菜品项目,还包括子菜单项目,子菜单也包含一系列菜品或者子菜单。

我们现在想实现两个个需求:

  • 如果是菜单项目,我们需要打印菜单的名称和描述,添加删除子菜单或者菜品,打印所有子菜单、子菜单包括的菜品、子菜单的子菜单的名称和描述,一直递归打印到最后一个菜品项目。
  • 如果是菜品项目,我们需要得到菜品的价格、描述、名称、是否是素菜这些信息

可以发现上述两个需求有相同和不同的地方,常规做法就是区别对待两者各自进行操作。但是这样以后扩展起来就非常麻烦,如果添加或者删除两者,那么原有代码就需要做相应的修改。而且两者其实很多操作都是类似的,却要写两套代码,操作繁琐。

分析下上面的图,我们不难发现这是一个典型的树形结构图,菜品项目是叶节点,子菜单项目是子节点(子节点还可以包含子节点或者叶节点),所有菜单是根节点,这个结构可以无限延伸下去。

如果我们能统一对待叶节点和子节点,使用一致的方式在树结构间游走处理,那就方便许多,这样不管以后新加一个叶节点还是子节点,原有代码都不需要修改,因为他们二者的处理方式完全一致。

下面我们就来看看具体的实现。


代码实现

1、定义叶节点和子节点的父类

先定义一个抽象类,作为叶节点和子节点的父类,父类定义了两者的所有操作方法,两者可以自己选择实现自己需要的方法。父类的每个方法默认实现都是抛出异常,等待子类覆盖实现。如果子类没有覆盖,然后又调用了该方法,就会抛出异常。

#import <Foundation/Foundation.h>

@interface MenuComponent : NSObject
-(void)add:(MenuComponent *)component;
-(void)remove:(MenuComponent *)component;
-(MenuComponent*)getChild:(NSInteger)position;
-(NSString*)getName;
-(NSString*)getDescription;
-(CGFloat)getPrice;
-(BOOL)isVegetarian;
-(void)print;
@end


===============
#import "MenuComponent.h"

@implementation MenuComponent
-(void)add:(MenuComponent *)component{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(void)remove:(MenuComponent *)component{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(MenuComponent *)getChild:(NSInteger)position{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(NSString *)getName{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(NSString *)getDescription{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(CGFloat)getPrice{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(BOOL)isVegetarian{
    NSString *reason = [NSString stringWithFormat:@"【%@】没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}

-(void)print{
    NSString *reason = [NSString stringWithFormat:@"%@没有实现该方法",NSStringFromClass([self class])];
    @throw ([NSException exceptionWithName:@"不支持该方法" reason:reason userInfo:nil]);
}


@end

2、实现菜单项目

#import "MenuComponent.h"

@interface Menu : MenuComponent
@property(copy ,nonatomic)NSString *name;
@property(copy ,nonatomic)NSString *desc;
@property(strong,nonatomic)NSMutableArray<MenuComponent *>* menuComponentArr;

-(instancetype)initMenuItemWithName:(NSString*)name withDesc:(NSString*)desc;
@end

========================================

#import "Menu.h"

@implementation Menu
-(instancetype)initMenuItemWithName:(NSString *)name withDesc:(NSString *)desc{
    if (self == [super init]) {
        self.name = name;
        self.desc = desc;
        self.menuComponentArr = [NSMutableArray array];
        
    }
    
    return self;
}

-(NSString *)getDescription{
    return self.desc;
}

-(NSString *)getName{
    return self.name;
}


-(void)add:(MenuComponent *)component{
    [self.menuComponentArr addObject:component];
}

-(void)remove:(MenuComponent *)component{
    [self.menuComponentArr enumerateObjectsUsingBlock:^(MenuComponent * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        if(obj == component){
            [self.menuComponentArr removeObject:component];
        }else{
            if ([obj isKindOfClass:[Menu class]]) {
                if ([((Menu *)obj).menuComponentArr containsObject:component]) {
                    [obj remove:component];
                    }
                }
            }
    }];
    
    
}


-(MenuComponent*)getChild:(NSInteger)position{
    return self.menuComponentArr[position];
}


-(void)print{
    NSLog(@"菜单名称:%@ | 菜单描述:%@ " ,self.name, self.desc);
    if(self.menuComponentArr.count){
        for (MenuComponent * component in self.menuComponentArr) {
            [component print];
        }
    }
}


@end

3、实现菜品项目

#import "MenuComponent.h"

@interface menuItem : MenuComponent
@property(copy ,nonatomic)NSString *name;
@property(copy ,nonatomic)NSString *desc;
@property(assign,nonatomic)NSInteger isVegetarain;
@property(assign,nonatomic)CGFloat price;

-(instancetype)initMenuItemWithName:(NSString*)name withDesc:(NSString*)desc withVegetarain:(NSInteger)isVege withPrice:(CGFloat)price;

@end

=====================================================

#import "menuItem.h"

@implementation menuItem
-(instancetype)initMenuItemWithName:(NSString *)name withDesc:(NSString *)desc withVegetarain:(NSInteger)isVege withPrice:(CGFloat)price{
    if (self == [super init]) {
        self.name = name;
        self.desc = desc;
        _isVegetarain = isVege;
        self.price = price;
        
    }
    
    return self;
}

-(CGFloat)getPrice{
    return self.price;
}

-(NSString *)getDescription{
    return self.desc;
}

-(NSString *)getName{
    return self.name;
}

-(BOOL)isIsVegetarain{
    return self.isVegetarain;
}

-(void)print{
    NSLog(@"菜品名称:%@ | 菜品价格:%f | 菜品描述:%@ | 是否是素菜:%@" ,self.name, self.price, self.desc, self.isVegetarain ? @"是":@"不是");
}

@end

4、客户端调试

我们先按照文章开头的图完成菜单的构建

MenuComponent *pancakeHouseMenu = [[Menu alloc]initMenuItemWithName:@"博饼屋菜单" withDesc:@"早餐"];
        MenuComponent *dinnerMenu = [[Menu alloc]initMenuItemWithName:@"正餐菜单" withDesc:@"午餐"];
        MenuComponent *cafeMenu = [[Menu alloc]initMenuItemWithName:@"咖啡菜单" withDesc:@"晚餐"];
        MenuComponent *dessertMenu = [[Menu alloc]initMenuItemWithName:@"甜点菜单" withDesc:@"饭后甜点"];
        MenuComponent *allMenu = [[Menu alloc]initMenuItemWithName:@"所有菜单" withDesc:@"所有菜单的组合"];
        
        [allMenu add:pancakeHouseMenu];
        [allMenu add:dinnerMenu];
        [allMenu add:cafeMenu];
        
        menuItem *meatItem = [[menuItem alloc]initMenuItemWithName:@"红烧肉" withDesc:@"祖传红烧肉,肥而不腻" withVegetarain:0 withPrice:177.2f];
        menuItem *fishItem = [[menuItem alloc]initMenuItemWithName:@"清蒸鲈鱼" withDesc:@"新鲜味美,回味无穷" withVegetarain:0 withPrice:2332.0f];
        [dinnerMenu add:meatItem];
        [dinnerMenu add:fishItem];
        
        menuItem *dessertItem1 = [[menuItem alloc]initMenuItemWithName:@"清炒小白菜" withDesc:@"味美而鲜,有机绿色无污染" withVegetarain:1 withPrice:17.3f];
        menuItem *dessertItem2 = [[menuItem alloc]initMenuItemWithName:@"玉米排骨汤" withDesc:@"饭后一口汤,快乐似神仙" withVegetarain:1 withPrice:243.3f];
        [dessertMenu add:dessertItem1];
        [dessertMenu add:dessertItem2];
        [dinnerMenu add:dessertMenu];
        

此时我们打印一下所有菜单,只需要一句命令就可以打印出所有的菜品项目和子菜单项目

[allMenu print];

输出如下:

2016-12-04 19:22:12.569 组合模式[39987:657657] 菜单名称:所有菜单 | 菜单描述:所有菜单的组合 
2016-12-04 19:22:12.570 组合模式[39987:657657] 菜单名称:博饼屋菜单 | 菜单描述:早餐 
2016-12-04 19:22:12.570 组合模式[39987:657657] 菜单名称:正餐菜单 | 菜单描述:午餐 
2016-12-04 19:22:12.570 组合模式[39987:657657] 菜品名称:红烧肉 | 菜品价格:177.199997 | 菜品描述:祖传红烧肉,肥而不腻 | 是否是素菜:不是
2016-12-04 19:22:12.570 组合模式[39987:657657] 菜品名称:清蒸鲈鱼 | 菜品价格:2332.000000 | 菜品描述:新鲜味美,回味无穷 | 是否是素菜:不是
2016-12-04 19:22:12.570 组合模式[39987:657657] 菜单名称:甜点菜单 | 菜单描述:饭后甜点 
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜品名称:清炒小白菜 | 菜品价格:17.299999 | 菜品描述:味美而鲜,有机绿色无污染 | 是否是素菜:是
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜品名称:玉米排骨汤 | 菜品价格:243.300003 | 菜品描述:饭后一口汤,快乐似神仙 | 是否是素菜:是
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜单名称:咖啡菜单 | 菜单描述:晚餐 

此时我们试着删除dessertMenu,然后再次打印菜单

2016-12-04 19:22:12.571 组合模式[39987:657657] 菜单名称:所有菜单 | 菜单描述:所有菜单的组合 
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜单名称:博饼屋菜单 | 菜单描述:早餐 
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜单名称:正餐菜单 | 菜单描述:午餐 
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜品名称:红烧肉 | 菜品价格:177.199997 | 菜品描述:祖传红烧肉,肥而不腻 | 是否是素菜:不是
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜品名称:清蒸鲈鱼 | 菜品价格:2332.000000 | 菜品描述:新鲜味美,回味无穷 | 是否是素菜:不是
2016-12-04 19:22:12.571 组合模式[39987:657657] 菜单名称:咖啡菜单 | 菜单描述:晚餐 

可以看到移除成功。

如果我们试着对dinnerMenu这个子菜单项目调用如下方法

[dinnerMenu isVegetarian];

会发现直接崩溃报错如下:

2016-12-04 20:40:44.049 组合模式[40371:710191] *** Terminating app due to uncaught exception '不支持该方法', reason: '【Menu】没有实现该方法'

此处就涉及到一个取舍问题:透明性和安全性谁更重要?

上面的例子就是保证了透明性,让子节点和叶节点被统一对待,如果调用了二者不支持的方法就直接抛出异常。安全性就需要对调用者做一个判断,如果调用者调用了错误的方法就不执行,这样保证不会抛出异常,但是需要区别调用者。

我们使用组合模式的意图就是为了保持叶节点和子节点的一致性,所以一般更偏重于透明性而不是安全性。

通过上面的例子大家应该对组合模式有了一个感性的认识,那么现在我们来具体看看组合模式的定义


定义

将 对 象 组 合 成 树 形 结 构 以 表 示 “ 部 分 -整 体 ” 的 层 次 结 构 。 组合模式 使 得 用 户 对 单 个 对 象 和组合对象的使用具有一致性。

组合模式的目的就是让客户端不用区分操作的对象是子节点还是叶节点,而是用一种统一的方式来操作。

实现这个目标的关键在于,设计一个抽象的组件类,让它可以代码子节点和叶节点,这样客户端就不需要区分二者,统一操作它们即可。

通常,组合模式都是用树形结构来表示的,通过根节点、子节点、叶节点组合成一颗对象树,这也意味着任何可以使用对象树来描述或者操作的功能,都可以使用组合模式来进行,比如XML解析,层次结构的菜单,iOS中由多个子视图构成的复杂视图,这些都可以使用组合模式来实现。

同时要注意,因为要让客户端统一操作子节点和叶节点,那么他们的抽象类就必须定义包含二者的所有方法,如果调用者调用了它们不支持的方法,可以抛出异常警告。这虽然违反了单一原则,但是在实际开发中是合理的。


好处

  • 定义了包含基本对象和组合对象的类层次结构

    基本对象可以被组合成更复杂的组合对
    象,而这个组合对象又可以被组合,这样不断的递归下去。客户代码中,任何用到基本
    对象的地方都可以使用组合对象。

  • 简化客户代码

    客户可以一致地使用组合结构和单个对象。通常用户不知道 (也不关心)处理的是一个叶节点还是一个组合组件。这就简化了客户代码 , 因为在定义组合的那些类中不需要写一些充斥着选择语句的函数。

  • 使 得 更 容 易 增 加 新 类 型 的 组 件

    新定义的 C o m p o s i t e 或 L e a f 子 类 自 动 地 与 已 有 的 结 构 和 客
    户代码一起工作,客户程序不需因新的 C o m p o n e n t类而改变。

  • 使你的设计变得更加一般化

    容易增加新组件也会产生一些问题,那就是很难限制组合
    中的组件。有时你希望一个组合只能有某些特定的组件。使用 C o m p o s i t e 时,你不能依赖类型系统施加这些约束,而必须在运行时刻进行检查。


使用时机

  • 如果你想表示对象的部分--整体层次结构。可以选用组合模式把整体和部分统一起来,使得层次结构实现更简单,从外部使用这个层次结构也更容易。

  • 如果你希望统一的使用组合结构中的所有对象,这正是组合模式提供的主要功能


Demo下载

组合模式Demo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容