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())
}
相关推荐
程序员张339 分钟前
SpringBoot计时一次请求耗时
java·spring boot·后端
Small black human4 小时前
设计模式-应用分层
设计模式
程序员岳焱7 小时前
Java 与 MySQL 性能优化:Java 实现百万数据分批次插入的最佳实践
后端·mysql·性能优化
麦兜*7 小时前
Spring Boot启动优化7板斧(延迟初始化、组件扫描精准打击、JVM参数调优):砍掉70%启动时间的魔鬼实践
java·jvm·spring boot·后端·spring·spring cloud·系统架构
大只鹅8 小时前
解决 Spring Boot 对 Elasticsearch 字段没有小驼峰映射的问题
spring boot·后端·elasticsearch
ai小鬼头8 小时前
AIStarter如何快速部署Stable Diffusion?**新手也能轻松上手的AI绘图
前端·后端·github
IT_10248 小时前
Spring Boot项目开发实战销售管理系统——数据库设计!
java·开发语言·数据库·spring boot·后端·oracle
bobz9658 小时前
动态规划
后端
stark张宇9 小时前
VMware 虚拟机装 Linux Centos 7.9 保姆级教程(附资源包)
linux·后端
亚力山大抵9 小时前
实验六-使用PyMySQL数据存储的Flask登录系统-实验七-集成Flask-SocketIO的实时通信系统
后端·python·flask