项目:咖啡计费系统

背景:现有系统中有一个抽象类Beverage,有2个抽象方法GetDescription和Cost。

 namespace DecoratorPattern
{
/// <summary>
/// 饮料抽象类
/// </summary>
public abstract class Beverage
{
protected string description = "饮料";
protected float price = 0f;
public abstract string GetDescription(); public abstract float Cost();
}
}

需求:目前有综合咖啡、深焙咖啡、浓缩咖啡,调料有牛奶、摩卡、豆浆、奶泡。未来可能增加新的咖啡种类和调料,当顾客点咖啡时,要求能够获得咖啡的描述和价格。

设计方案1:设计综合咖啡、深焙咖啡、浓缩咖啡4个子类,继承Beverage。再用这4个子类分别派生4个子类,带有牛奶的综合咖啡,带有摩卡的综合咖啡,带有豆浆的综合咖啡...

分析:缺点时显而易见的,这样做导致“类爆炸”,一共需要3*4=12个子类。

设计方案2:把调料作为咖啡的属性设置在Beverage里,并增加方法HasMilk(),SetMilk()等类似的方法。

分析:缺点这样做无疑时从一个灾难跳进另一个灾难中。我们在开发中应当尽量避免修改已有代码,遵循“开闭原则”。而且当增加新的饮料时,又要修改基类。

另一个灾难是,子类在计算价格时,需要大量的分支结构来判断是否包含某种调料,以计算咖啡的价格,我们总是尽量的避免复杂的分支结构,这使得维护变得非常困难。

还有针对实现编程带来的问题,不能够动态的添加职责。

装饰者模式 Decorator 闪亮登场:

装饰者模式动态的将职责附加到对象上。若要扩展共呢个,装饰者提供了比继承更有弹性的替代方案。

1. 4个基类继承Beverage

 namespace DecoratorPattern
{
/// <summary>
/// 综合咖啡
/// </summary>
public class HouseBlend:Beverage
{
public HouseBlend(float price)
{
this.price = price;
this.description = "综合咖啡";
} public override float Cost()
{
return price;
}
public override string GetDescription()
{
return this.description;
}
}
}
 namespace DecoratorPattern
{
/// <summary>
/// 深焙咖啡
/// </summary>
public class DarkRoast:Beverage
{
public DarkRoast(float price)
{
this.price = price;
this.description = "深焙咖啡";
}
public override string GetDescription()
{
return this.description;
}
public override float Cost()
{
return price;
}
}
}
 namespace DecoratorPattern
{
/// <summary>
/// 浓缩咖啡
/// </summary>
public class Espresso:Beverage
{
public Espresso(float price)
{
this.price = price;
this.description = "浓缩咖啡";
}
public override float Cost()
{
return this.price;
}
public override string GetDescription()
{
return this.description;
}
}
}

装饰者继承Beverage,注意这里继承的目的并不是为了获得基类的功能,而是为了类型匹配,达到多态的目的,获得功能由组合来实现。因此每一个装饰者都需要维护一个Beverage引用。

 namespace DecoratorPattern
{
/// <summary>
/// 摩卡装饰者,为了能够取代Beverage,所以CondimentDecorator继承自Beverage,目的并非获得Beverage
/// 而是为了类型匹配
/// </summary>
public class Mocha : Beverage
{
//持有抽象类饮料的引用,达到运行时添加职责的目的
Beverage beverage;
//包装Beverage
public Mocha(Beverage b, float price)
{
beverage = b;
this.price = price;
}
public override float Cost()
{
return beverage.Cost() + price;
} public override string GetDescription()
{
return beverage.GetDescription() + ", 摩卡";
}
}
}
 namespace DecoratorPattern
{
/// <summary>
/// 奶泡装饰者
/// </summary>
public class Whip : Beverage
{
private Beverage beverage;
public Whip(Beverage b, float price)
{
beverage = b;
this.price = price;
}
public override float Cost()
{
return beverage.Cost() + price;
}
public override string GetDescription()
{
return (beverage.GetDescription() + " ,奶泡");
}
}
}
 namespace DecoratorPattern
{
/// <summary>
/// 豆浆装饰者
/// </summary>
public class Soy : Beverage
{
private Beverage beverage;
public Soy(Beverage b, float price)
{
this.price = price;
beverage = b;
}
public override float Cost()
{
return beverage.Cost() + price;
} public override string GetDescription()
{
return( beverage.GetDescription() + ", 豆浆");
}
}
}

客户端类:CoffeeShop.cs

 using System;

 namespace DecoratorPattern
{
class CoffeeShop
{
static void Main(string[] args)
{
//来一杯浓缩咖啡,不要调料
Beverage beverage = new Espresso(1.99f);
Console.WriteLine(beverage.GetDescription() + "$" + beverage.Cost()); //来一杯摩卡奶泡深焙咖啡
Beverage beverage2 = new DarkRoast(0.99f);
beverage2 = new Whip(beverage2, 0.1f); //用奶泡装饰深焙咖啡
beverage2 = new Mocha(beverage2, 0.2f); //再用摩卡装饰
Console.WriteLine(beverage2.GetDescription() + "$" + beverage2.Cost()); Console.ReadKey();
} }
}

运行结果:

参考资料《Head First 设计模式》

装饰者模式 Decorator的更多相关文章

  1. 浅谈设计模式--装饰者模式(Decorator Pattern)

    挖了设计模式这个坑,得继续填上.继续设计模式之路.这次讨论的模式,是 装饰者模式(Decorator Pattern) 装饰者模式,有时也叫包装者(Wrapper),主要用于静态或动态地为一个特定的对 ...

  2. 【PHP设计模式 09_ZhuangShiQi.php】装饰器模式 (decorator)

    <?php /** * [装饰器模式 (decorator)] * 有时候发布一篇文章需要经过很多人手,层层处理 */ header("Content-type: text/html; ...

  3. 设计模式(八)装饰器模式Decorator(结构型)

    设计模式(八)装饰器模式Decorator(结构型) 1. 概述 若你从事过面向对象开发,实现给一个类或对象增加行为,使用继承机制,这是所有面向对象语言的一个基本特性.如果已经存在的一个类缺少某些方法 ...

  4. 设计模式 - 装饰者模式(Decorator Pattern) Java的IO类 用法

    装饰者模式(Decorator Pattern) Java的IO类 用法 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26716 ...

  5. 设计模式 - 装饰者模式(Decorator Pattern) 具体解释

    装饰者模式(Decorator Pattern) 具体解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26707033 装饰者 ...

  6. 装饰器模式-Decorator(Java实现)

    装饰器模式-Decorator(Java实现) 装饰器模式允许向一个现有的对象添加新的功能, 同时又不改变其结构. 其中 "现有对象"在本文中是StringDisplay类. 添加 ...

  7. 设计模式学习--装饰者模式(Decorator Pattern)

    概念: 装饰者模式(Decorator Pattern): 动态地将功能添加到对象,相比生成子类更灵活,更富有弹性. 解决方案: 装饰者模式的重点是对象的类型,装饰者对象必须有着相同的接口,也也就是有 ...

  8. 大话设计模式--装饰者模式 Decorator -- C++实现实例

    1.装饰者模式 Decorator 动态地给一个对象添加一个额外的职责, 就添加功能来说, 装饰模式比生成子类更为灵活. 每个装饰对象的实现和如何使用这个对象分离,  每个装饰对象只关心自己的功能,不 ...

  9. 设计模式(三):“花瓶+鲜花”中的装饰者模式(Decorator Pattern)

    在前两篇博客中详细的介绍了"策略模式"和“观察者模式”,今天我们就通过花瓶与鲜花的例子来类比一下“装饰模式”(Decorator Pattern).在“装饰模式”中很好的提现了开放 ...

随机推荐

  1. Windows Vue 安装

    https://nodejs.org/dist/v6.9.5/node-v6.9.5-x64.msi 新建文件夹 node_global新建文件夹 node_cachenpm config set p ...

  2. [jzoj]1115.【HNOI2008】GT考试

    Link https://jzoj.net/senior/#main/show/1115 Description 申准备报名参加GT考试,准考证号为n位数X1X2X3...Xn-1Xn(0<=X ...

  3. SoapUI破解及安装教程

    之前学了一段时间的SoapUI,但是好久不用了,这里记录下专业版的破解的流程,后续的学习会不断更新. soapUI安装及破解(这里针对专业版) 下载地址:http://dl.eviware.com/l ...

  4. jquery提示sucess

    这是学习笔记. 今天做东西的时候,想把体验做好,于是打算再ajax success字段中添加函数实现提示sucess. 用了jquery的fadeIn 跟fadeOut,再fadeIn的callbac ...

  5. (91)Wangdao.com第二十四天_Mutation Observer API 突变监视器

    Mutation Observer API 突变监视接口 用来监视 DOM 变动. DOM 的任何变动,比如节点的增减.属性的变动.文本内容的变动,这个 API 都可以得到通知 概念上,它很接近事件, ...

  6. vue菜鸟从业记:公司项目里如何进行前后端接口联调

    最近我的朋友王小闰进入一家新的公司,正好公司项目采用的是前后端分离架构,技术栈是王小闰非常熟悉的vue全家桶,后端用的是Java语言. 在前后端开发人员碰面之后,协商确定好了前端需要的数据接口(扯那么 ...

  7. 反射RelectionDemo

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...

  8. 手写AVL 树(下)

    上一篇 手写AVL树上实现了AVL树的插入和查询 上代码: 头文件:AVL.h #include <iostream> template<typename T1,typename T ...

  9. java.text.DateFormat 线程不安全问题

    java.text下的 DateFormat 是线程不安全的: 建议1: 1.使用threadLocal包装DateFormat(太复杂,不推荐) 2.使用org.apache.commons.lan ...

  10. Multi-Projector Based Display Code ---- Calibration

    Overview As mentioned previously, there are two main steps in generating a seamless display. The fir ...