设计模式之装饰者模式

作者: 纳萨立克 | 来源:发表于2017-03-21 14:29 被阅读195次

设计模式之装饰者模式

需求场景

咖啡店订单系统

需求分析

咖啡种类比较多,结账的时候我们需要知道咖啡名和价格。首先想到的是我们创建一个咖啡的基类,该类提供一个价格和名称的接口,店里的所有的饮料都继承该类。

问题

  • 我们给每种咖啡都要创建一个子类,一个咖啡店有很多种饮料,这样就会创建很多的类,类的数目过于庞大
  • 当咖啡店不断出新品的时候,系统不便于维护

分析

决定价格的因素:

  • 不同种类的咖啡的价格不一样
  • 咖啡里面的加入的调料的价格也不同

咖啡饮料的价格 = 咖啡的价格 + 加入调料的价格

由上边可以看出:

咖啡相当于‘被装饰者’ ,调料相当于’装饰者‘
例如:
Espresso Macchiato(浓缩玛奇朵 = Espresso(浓缩咖啡) +Milk(牛奶) + Mocha(摩卡)。

Espresso 相当于”被装饰者“ , Mocha和Milk相当于‘装饰者’

只要对被装饰者和装饰者进行不同的组合,就可以得到不同的咖啡

这样的组合是动态的,被装饰者和装饰者不是写死在类里的比如继承,类继承是在编译的时候增加行为,而装饰者模式是在运行时增加行为),而是动态的组合,即在运行时进行绑定

设计结构如下图:

img

客点了一杯 Espresso Macchiato(浓缩玛奇朵),那么系统将会开始以下的工作流程:

  • 首先实例化一个Espresso对象,对象里面有名字和基本价格
  • 实例化一个milk的装饰者对象,对象里面有名字和价格,同时让milk持有Espresso
  • 调用milk的cost()方法,这是会调用Espresso的cost方法,并把返回的价格和milk的价格相加。
  • 实例化一个Mocha的装饰者对象,对象里面有名字和价格,同时让Mocha持有milk对象
  • 最后调用 Mocha 对象的 cost() 方法,这个方法会去调用 Milk 对象的 cost() 方法,并将返回的价格和 mocha 的价格相加,如此我们就得到了 Espresso 配 milk 和 mocha 的价格。

代码实现

Beverage.h
#import <Foundation/Foundation.h>



@protocol Beverage <NSObject>

@optional


-(NSString *)getName;

-(double)cost;


@end

CondimentDecorator.h
#import <Foundation/Foundation.h>
#import "Beverage.h"

@protocol CondimentDecorator <Beverage>

@end

Espresso.h
#import <Foundation/Foundation.h>
#import "Beverage.h"
@interface Espresso : NSObject<Beverage>

@end

Espresso.m
#import "Espresso.h"

@implementation Espresso{

    NSString * _name;
}


-(instancetype)init{

    if(self = [super init]){
    
        _name = @"Espresso";
    }
    
    return self;
}

-(NSString *)getName{

    return _name;
}

-(double)cost{

    return 10.0;
}

@end

Milk.h
#import <Foundation/Foundation.h>
#import "Beverage.h"
#import "CondimentDecorator.h"

@interface Milk : NSObject<CondimentDecorator>

@property(nonatomic,retain) id<Beverage> Beverage;


-(instancetype)initWithBeverage:(id<Beverage>)beverage;

@end
Milk.m

#import "Milk.h"
#import "Beverage.h"


@implementation Milk{

    NSString * _name;
}


-(instancetype)initWithBeverage:(id<Beverage>)beverage{

    if(self = [super init]){
        
        _name = @"Milk";
    
        self.Beverage = beverage;
    
    }
    
    return self;
    
}


-(NSString *)getName{

    return [NSString stringWithFormat:@"%@ + %@",[self.Beverage getName],_name];
}

-(double)cost{

    return [self.Beverage cost] + 4.0;
}


@end

Mocha同Milk

调用

    Espresso * ex = [[Espresso alloc] init];
    
    NSLog(@"%@ %.2f",ex.getName,ex.cost);
    
    Milk * milk = [[Milk alloc] initWithBeverage:ex];
    
    NSLog(@"%@ %.2f",milk.getName,milk.cost);
    
    Mocha * mocha = [[Mocha alloc] initWithBeverage:milk];
    
    NSLog(@"%@ %.2f",mocha.getName,mocha.cost);

参考文章:
《HeadFirst设计模式》

相关文章

  • JavaScript 设计模式核⼼原理与应⽤实践 之 结构型设计

    JavaScript 设计模式核⼼原理与应⽤实践 之 结构型设计模式 装饰器模式,又名装饰者模式。它的定义是“在不...

  • Java设计模式之 —— 装饰者(Decorator) — 点

    Java设计模式之 —— 装饰者(Decorator) — 点炒饭 下面会用做炒饭的例子来描述一下装饰者设计模式,...

  • 设计模式

    设计模式 单例模式、装饰者模式、

  • 设计模式笔记汇总

    目录 设计原则 “依赖倒置”原则 未完待续... 设计模式 设计模式——策略模式 设计模式——装饰者模式 设计模式...

  • java IO 的知识总结

    装饰者模式 因为java的IO是基于装饰者模式设计的,所以要了解掌握IO 必须要先清楚什么事装饰者模式(装饰者模式...

  • 设计模式

    常用的设计模式有,单例设计模式、观察者设计模式、工厂设计模式、装饰设计模式、代理设计模式,模板设计模式等等。 单例...

  • 设计模式之装饰者模式

    如果你没了解过装饰者模式,那么请继续往下看,如果你是老司机,那么,你可以快速往下看。 开始装个13,再进入正文。 ...

  • 设计模式之装饰者模式

    该模式可以避免滥用继承,在使用对象组合的方式,就能做到在运行时装饰类,此后便能在不修改任何底层代码的情况下给对象赋...

  • 设计模式之装饰者模式

    装饰者模式的定义是动态地给一个对象添加一些额外的职责。就增加功能来说,装饰者模式比生成子类更加灵活。 通常给一个对...

  • 设计模式之装饰者模式

    前言: 相信Java开发者在使用java i/o API的时候,对于以下代码写法应该非常熟悉: 在不知道具体设计实...

网友评论

本文标题:设计模式之装饰者模式

本文链接:https://www.haomeiwen.com/subject/bmhgnttx.html