GO设计模式——11、装饰器模式(结构型)

目录

[装饰器模式(Decorator Pattern)](#装饰器模式(Decorator Pattern))

装饰器模式的核心角色:

优缺点

使用场景

代码实现


装饰器模式(Decorator Pattern)

装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。装饰器模式通过将对象包装在装饰器类中,以便动态地修改其行为。这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能。

装饰器模式的核心角色

  • 抽象组件(Component):定义了对象的接口,可以是一个抽象类或接口。
  • 具体组件(ConcreteComponent):实现了抽象组件的接口,是被装饰的对象。
  • 装饰器(Decorator):维持一个指向抽象组件对象的引用,并实现了抽象组件的接口。
  • 具体装饰器(ConcreteDecorator):具体的装饰器对象,用于扩展具体组件的功能。

优缺点

(1)优点:

  • 装饰类和被装饰类可以独立发展,不会相互耦合,装饰模式是继承的一个替代模式,装饰模式可以动态扩展一个实现类的功能。

(2)缺点:多层装饰比较复杂。

使用场景

  • 扩展一个类的功能。
  • 动态增加功能,动态撤销。

代码实现

Go 复制代码
package main

import (
    "fmt"
)

// 抽象组件:咖啡接口
type Coffee interface {
    GetDescription() string
    GetCost() float64
}

// 具体组件:原味咖啡
type PlainCoffee struct{}

func (c *PlainCoffee) GetDescription() string {
    return "Plain Coffee"
}

func (c *PlainCoffee) GetCost() float64 {
    return 1.0
}

// 装饰器:配料装饰器
type IngredientDecorator struct {
    coffee    Coffee
    extraCost float64
    extraDesc string
}

func (d *IngredientDecorator) GetDescription() string {
    return d.coffee.GetDescription() + ", " + d.extraDesc
}

func (d *IngredientDecorator) GetCost() float64 {
    return d.coffee.GetCost() + d.extraCost
}

// 具体装饰器:牛奶装饰器
type MilkDecorator struct {
    IngredientDecorator
}

func NewMilkDecorator(coffee Coffee) *MilkDecorator {
    return &MilkDecorator{
       IngredientDecorator{
          coffee:    coffee,
          extraCost: 0.5,
          extraDesc: "Milk",
       },
    }
}

// 具体装饰器:糖浆装饰器
type SyrupDecorator struct {
    IngredientDecorator
}

func NewSyrupDecorator(coffee Coffee) *SyrupDecorator {
    return &SyrupDecorator{
       IngredientDecorator{
          coffee:    coffee,
          extraCost: 0.3,
          extraDesc: "Syrup",
       },
    }
}

func main() {
    coffee := &PlainCoffee{}
    fmt.Println("Coffee:", coffee.GetDescription(), "Cost:", coffee.GetCost())

    coffeeWithMilk := NewMilkDecorator(coffee)
    fmt.Println("Coffee with Milk:", coffeeWithMilk.GetDescription(), "Cost:", coffeeWithMilk.GetCost())

    coffeeWithMilkAndSyrup := NewSyrupDecorator(coffeeWithMilk)
    fmt.Println("Coffee with Milk and Syrup:", coffeeWithMilkAndSyrup.GetDescription(), "Cost:", coffeeWithMilkAndSyrup.GetCost())
}
相关推荐
Victor3569 分钟前
MongoDB(109)如何使用Robo 3T?
后端
鹏程十八少9 分钟前
9. 2026金三银四 面试官问不垮:Java VS Android 设计模式 16 讲
前端·后端·面试
skilllite作者13 分钟前
从“记忆”到“项目 Wiki”:我在 SkillLite 里实现了一套 Markdown-only LLM Wiki 自动维护机制
开发语言·jvm·人工智能·后端·架构·rust
Victor35615 分钟前
MongoDB(110)什么是MongoDB Atlas?
后端
jieyucx17 分钟前
Go 语言函数入门:定义、参数、返回值
c++·算法·golang·入门·函数
XMYX-020 分钟前
20 - Go 互斥锁:Mutex 与并发安全
开发语言·golang
geovindu35 分钟前
go: Observer Pattern
开发语言·观察者模式·设计模式·golang
奇逍科技圈35 分钟前
开源架构 + BC 一体化:批发零售企业订货系统源码重构增长新路径
后端·架构·开源
代码羊羊38 分钟前
Rust Panic 深入全解:不可恢复错误的处理与原理
开发语言·后端·rust
会编程的土豆44 分钟前
从 C/C++ 视角快速上手 Go 语言:核心差异与避坑指南
c语言·开发语言·c++·后端·golang