装饰者模式

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。



7
第7章 设计模式

7.3 装饰者模式

设计原则(Open-Closed Principle) 类应该对扩展开放,对修改关闭。 我们的目标是允许类容易扩展,在不修改现有代码的情况 下,就可搭配新的行为。如能实现这样的目标,其好处在 于:这样的设计具有弹性,可以应对改变,可以接受新的 功能来应对改变的需求。 如何让设计的每个部分都遵循开放-关闭原则?
当哪些因素改变时会影响这个设计?
调料价钱的改变会使我们更改现有代码。 一旦出现新的调料,我们就需要加上新的方法,并改变 超类中的cost()方法。 以后可能会开发出新饮料,对于这些饮料而言(冰茶), 某些调料可能并不适合,但是在这个设计方式中,Tea 子类仍将继承那些不合适的方法,比如:hasWhip()。 如何顾客想要双倍摩卡,怎么办? ……
27
16
第7章 设计模式

7.3 装饰者模式
问题:如果有一张订单:“双倍摩卡豆浆奶泡拿铁咖
啡”,应该如何进行设计?
17
第7章 设计模式

7.3 装饰者模式
Beverage类

Condiment类
18
第7章 设计模式

7.3 装饰者模式
饮料(被装饰者)的代码
19
第7章 设计模式

7.3 装饰者模式

7.3 装饰者模式
实现1 类爆炸

调料的增加/删除 调料价格的修改
3
第7章 设计模式

7.3 装饰者模式
实现2

使用实例变量和继承,跟踪这些调料
4
第7章 设计模式

7.3 装饰者模式
实现2
5
第7章 设计模式

7.3 装饰者模式
问题:实现cost方法
6
第7章 设计模式


7.3 装饰者模式
9
第7章 设计模式
1.
7.3 装饰者模式
以DarkRoast对象开始
2.
顾客需要摩卡(Mocha),所以建立一个Mocha对象,并用 它将DarkRoast对象包起来
10
第7章 设计模式
3.
7.3 装饰者模式
顾客也想要奶泡(Whip),所以建立一个Whip装饰者,并 用它将Mocha对象包起来
11

装饰者和被装饰者必须是一样的类型,我们在此使用继 承达到“类型匹配” 类型匹配意味着装饰者和被装饰者具有相同的接口,从 而装饰者可以取代被装饰者 新的行为并不是继承自超类,而是由组合对象得到,即 所有饮料和调料可以更有弹性的加以混合和匹配 我们可以在任何时候,实现新的装饰者增加新的行为。 如果依赖继承,每当需要新行为时,必须修改代码 Component类型可以使用抽象类,也可以使用接口
8
第7章 设计模式


7.3 装饰者模式
星巴克咖啡订单管理系统——使用装饰者模式
以饮料(Beverage)为主体,然后在运行时以调料 (Condiment)来装饰(decorate)饮料
比如,顾客想要摩卡和奶泡深焙咖啡,那么:



取出一个深焙咖啡(DarkRoast)对象 以摩卡(Mocha)对象装饰它 以奶泡(Whip)对象装饰它 调用cost方法,并依赖委托(delegrate)将调料的价格加上去
第7章 设计模式
4.
7.3 装饰者模式
通过调用最外圈装饰者(Whip)的cost即可计算饮料价格。
12
第7章 设计模式

7.3 装饰者模式
“装饰者模式”——特点

来自百度文库

装饰者和被装饰对象具有相同的超类型 可以用一个或多个装饰者包装一个对象 由于装饰者和被装饰对象有相同的超类型,所以在任何需要原始对 象(被包装)的场合,都可以用装饰过的对象代替她 装饰者可以在所委托被装饰者的行为之前或之后,加上自己的行为, 以达到特定的目的 对象可以在任何时候被装饰,所以可以在运行时动态的、不限量的 用需要的装饰者来装饰对象
24
第7章 设计模式

7.3 装饰者模式
问题1 星巴克决定在菜单上加上咖啡的容量大小,供顾 客选择小杯(tall)、中杯(grande)、大杯(venti);另 外还需要根据咖啡容量收费,比如小中大杯咖啡 加上豆浆(Soy),分别加收0.10、0.15、0.20美金。 我们应该如何改变装饰者类应对这样的需求?
饮料(被装饰者)的代码
20
第7章 设计模式

7.3 装饰者模式
调料(装饰者)的代码
21
第7章 设计模式

7.3 装饰者模式
测试代码
22
第7章 设计模式

7.3 装饰者模式
测试结果
23
第7章 设计模式

7.3 装饰者模式
总结



在我们的代码中,应该允许行为可以被扩展,而无需修改现 有的代码 装饰者模式意味着一群装饰者类,这些类用来包装具体组件 装饰者反映出被装饰者的组件类型(具有相同的类型) 装饰者可以在被装饰者的行为前面或后面加上自己的行为, 甚至将被装饰者的行为整个取代掉,而达到特定的目的 可以用无数个装饰者包装一个组件 装饰者一般对组件的客户是透明的,除非客户程序依赖于组 件的具体类型 装饰者会导致设计中出现许多小对象,如果过度使用,会让 程序变得复杂


这通常是无法做到的。要让OO设计同时具有开放性和关闭性, 又不修改现有的代码,需要花费许多时间和努力。一般来说, 我们没有足够的精力把设计的每个部分都这么设计,这可能 只是一种浪费。 遵循开放-关闭原则,通常会引入新的抽象层次,增加代码的 复杂度。你需要把注意力集中在设计中最有可能改变的地方, 然后应用开放-关闭原则。
25
第7章 设计模式
7.3 装饰者模式
26
第7章 设计模式

7.3 装饰者模式
问题2 在装饰者模式中,是否可以允许装饰者知道一连 串装饰链条中其他装饰者地存在?比如,我们需 要使用getDescription()列出“Whip, Double Mocha”,而不是“Mocha, Whip, Mocha”。 我们应该如何设计装饰者模式才能满足这种需求?
第7章 设计模式

7.3 装饰者模式
星巴克咖啡订单管理系统 管理、计算各种饮料的售价。
Mocha 摩卡
Espresso 浓缩咖啡
DarkRoast 深焙咖啡
1
Milk 蒸奶
HouseBlend 家常咖啡
Ice 冰
Decaf 脱咖啡因咖啡
第7章 设计模式

7.3 装饰者模式
实现1
2
第7章 设计模式

装饰者模式定义 装饰者模式动态地将责任附加到对象上。若要扩展 功能,装饰者提供了比继承更有弹性的替代方案。
13
第7章 设计模式
7.3 装饰者模式
14
第7章 设计模式
7.3 装饰者模式
15
第7章 设计模式

7.3 装饰者模式
问题 为什么Decorate类扩展自Component类?


相关文档
最新文档