结构型模式-装饰器模式

什么是装饰器模式

装饰是一种结构型设计模式, 允许你通过将对象放入包含行为的特殊封装对象中来为原对象绑定新的行为

  1. 部件(Component)声明封装器和被封装对象的公用接口
  2. 具体部件(Concrete Component)类是被封装对象所属的类。 它定义了基础行为,但装饰类可以改变这些行为
  3. 基础装饰(Base Decorator)类拥有一个指向被封装对象的引 用成员变量。该变量的类型应当被声明为通用部件接口,这 样它就可以引用具体的部件和装饰。装饰基类会将所有操作 委派给被封装的对象
  4. 具体装饰类(Concrete Decorators)定义了可动态添加到部 件的额外行为。具体装饰类会重写装饰基类的方法,并在调 用父类方法之前或之后进行额外的行为
  5. 客户端(Client)可以使用多层装饰来封装部件,只要它能 使用通用接口与所有对象互动即可

Example

零件接口

1
2
3
4
5
package main

type IPizza interface {
getPrice() int
}

具体零件

1
2
3
4
5
6
7
8
package main

type VeggieMania struct {
}

func (p *VeggieMania) getPrice() int {
return 15
}

具体装饰a

1
2
3
4
5
6
7
8
9
10
package main

type TomatoTopping struct {
pizza IPizza
}

func (c *TomatoTopping) getPrice() int {
pizzaPrice := c.pizza.getPrice()
return pizzaPrice + 7
}

具体装饰b

1
2
3
4
5
6
7
8
9
10
package main

type CheeseTopping struct {
pizza IPizza
}

func (c *CheeseTopping) getPrice() int {
pizzaPrice := c.pizza.getPrice()
return pizzaPrice + 10
}

客户端

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package main

import "fmt"

func main() {

pizza := &VeggieMania{}

//Add cheese topping
pizzaWithCheese := &CheeseTopping{
pizza: pizza,
}

//Add tomato topping
pizzaWithCheeseAndTomato := &TomatoTopping{
pizza: pizzaWithCheese,
}

fmt.Printf("Price of veggeMania with tomato and cheese topping is %d\n", pizzaWithCheeseAndTomato.getPrice())
}

适用场景

  1. 如果你希望在无需修改代码的情况下即可使用对象,且希望 在运行时为对象新增额外的行为,可以使用装饰模式。
  • 装饰能将业务逻辑组织为层次结构,你可为各层创建一个装 饰,在运行时将各种不同逻辑组合成对象。由于这些对象都 遵循通用接口,客户端代码能以相同的方式使用这些对象
  1. 如果用继承来扩展对象行为的方案难以实现或者根本不可行, 你可以使用该模式。
  • 许多编程语言使用 final 最终 关键字来限制对某个类的进 一步扩展。复用最终类已有行为的唯一方法是使用装饰模式: 用封装器对其进行封装

优缺点

优点

  • 无需创建新子类即可扩展对象的行为
  • 可以在运行时添加或删除对象的功能
  • 可以用多个装饰封装对象来组合几种行为
  • 单一职责原则。你可以将实现了许多不同行为的一个大类拆 分为多个较小的类

缺点

  • 在封装器栈中删除特定封装器比较困难
  • 实现行为不受装饰栈顺序影响的装饰比较困难
  • 各层的初始化配置代码看上去可能会很糟糕

与其他模式的关系

  • 适配器可以对已有对象的接口进行修改,装饰则能在不改变 对象接口的前提下强化对象功能。此外,装饰还支持递归组 合,适配器则无法实现

  • 适配器能为被封装对象提供不同的接口,代理能为对象提供 相同的接口,装饰则能为对象提供加强的接口

    • 责任链和装饰模式的类结构非常相似。两者都依赖递归组合 将需要执行的操作传递给一系列对象。但是,两者有几点重 要的不同之处。责任链的管理者可以相互独立地执行一切操作,还可以随时 停止传递请求。另一方面,各种装饰可以在遵循基本接口的 情况下扩展对象的行为。此外,装饰无法中断请求的传递
  • 组合和装饰的结构图很相似,因为两者都依赖递归组合来组 织无限数量的对象

  • 大量使用组合和装饰的设计通常可从对于原型的使用中获益。 你可以通过该模式来复制复杂结构,而非从零开始重新构造

  • 装饰可让你更改对象的外表,策略则让你能够改变其本质

  • 装饰和代理有着相似的结构,但是其意图却非常不同。这两 个模式的构建都基于组合原则,也就是说一个对象应该将部 分工作委派给另一个对象。两者之间的不同之处在于代理通 常自行管理其服务对象的生命周期,而装饰的生成则总是由 客户端进行控制