设计模式——装饰者模式
创始人
2025-01-08 12:06:08
0

设计模式——装饰者模式

  • 1.问题
    • 1.1 方案一
    • 1.2 方案二
  • 2.装饰者模式
    • 2.1 基本介绍
    • 2.2 结构
    • 2.3 代码实现
  • 3.小结

1.问题

咖啡订单项目:

  1. 咖啡种类/单品咖啡:Espresso(意大利浓咖啡)、ShortBlack、LongBlack(美式咖啡)、Decaf(无因咖啡)
  2. 调料:Milk、Soy(豆浆)、Chocolate
  3. 要求在扩展新的咖啡种类时,具有良好的扩展性、改动方便、维护方便
  4. 使用 OO 的来计算不同种类咖啡的费用:客户可以点单品咖啡,也可以单品咖啡+调料组合。

1.1 方案一

方案一
问题解析:

  1. Drink 是一个抽象类,表示饮料
  2. des 就是对咖啡的描述,比如咖啡的名字
  3. cost()方法就是计算费用,Drink 类中做成一个抽象方法
  4. Decaf 就是单品咖啡, 继承 Drink,并实现 cost
  5. Espress && Milk 就是单品咖啡+调料,这个组合很多,容易产生过多的子类

1.2 方案二

方案二
这里的milk、soy、chocolate都是Boolean类型

问题解析:

  1. 方案2可以控制类的数量,不至于造成很多的类在增加或者删除调料种类时,代码的维护量很大
  2. 考虑到用户可以添加多份 调料时,可以将 hasMilk 返回一个对应 int
  3. 考虑使用 装饰者 模式

2.装饰者模式

2.1 基本介绍

  1. 装饰者模式:动态的将新功能附加到对象上。在对象功能扩展方面,它比继承更有弹性,装饰者模式也体现了开闭原则(ocp)
  2. 这里提到的动态的将新功能附加到对象和 ocp 原则

2.2 结构

装饰(Decorator)模式中的角色:

  • 抽象构件(Component)角色 :定义一个抽象接口以规范准备接收附加责任的对象。
  • 具体构件(Concrete Component)角色 :实现抽象构件,通过装饰角色为其添加一些职责。
  • 抽象装饰(Decorator)角色 : 继承或实现抽象构件,并包含具体构件的实例,可以通过其子
    类扩展具体构件的功能。
  • 具体装饰(ConcreteDecorator)角色 :实现抽象装饰的相关方法,并给具体构件对象添加附
    加的责任。

2.3 代码实现

类图展示:
核心
抽象类Drink:

public abstract class Drink {     private String description;     private double price;      public abstract double cost();      public String getDescription() {         return description;     }      public void setDescription(String description) {         this.description = description;     }      public double getPrice() {         return price;     }      public void setPrice(double price) {         this.price = price;     } } 

装饰者:

public class Decorator extends Drink {      private Drink drink;      public Decorator(Drink drink) {         this.drink = drink;     }      @Override     public double cost() {         return super.getPrice() + drink.cost();     }      @Override     public String getDescription() {         return super.getDescription() + " " + getPrice() + " && " + drink.getDescription();     } } 

Coffee:

public class Coffee extends Drink {      @Override     public double cost() {         return super.getPrice();     } } 

Coffee子类:

public class Decaf extends Coffee {     public Decaf() {         setDescription("无因咖啡");         setPrice(1);     } } public class Espresso extends Coffee {     public Espresso() {         setDescription("意大利咖啡");         setPrice(6);     } } public class LongBlack extends Coffee {     public LongBlack() {         setDescription("longBlack");         setPrice(5);     } } public class ShortBlack extends Coffee {     public ShortBlack() {         setDescription("shortBlack");         setPrice(4);     } } 

装饰器子类:

public class Soy extends Decorator {     public Soy(Drink drink) {         super(drink);         setDescription("豆浆");         setPrice(1.5);     } } public class Milk extends Decorator{     public Milk(Drink drink) {         super(drink);         setDescription("牛奶");         setPrice(2);     } } public class Chocolate extends Decorator {      public Chocolate(Drink drink) {         super(drink);         setDescription("巧克力");         setPrice(3);     } } 

Client调用:

public class Client {     public static void main(String[] args) {         // 两份巧克力+一份牛奶Longblack         Drink order1 = new LongBlack();         System.out.println("order1费用:" + order1.cost());         System.out.println("order1描述:" + order1.getDescription());          order1 = new Milk(order1);         System.out.println("order1加一份牛奶费用:" + order1.cost());         System.out.println("order1加一份牛奶描述:" + order1.getDescription());          order1 = new Chocolate(order1);         System.out.println("order1加一份牛奶,加一份巧克力费用:" + order1.cost());         System.out.println("order1加一份牛奶,加一份巧克力描述:" + order1.getDescription());          order1 = new Chocolate(order1);         System.out.println("order1加一份牛奶,加两份巧克力费用:" + order1.cost());         System.out.println("order1加一份牛奶,加两份巧克力描述:" + order1.getDescription());          System.out.println("==============================");           Drink order2 = new Decaf();         System.out.println("无因咖啡费用:" + order2.cost());         System.out.println("无因咖啡描述:" + order2.getDescription());          order2 = new Milk(order2);         System.out.println("无因咖啡,加一份牛奶费用:" + order2.cost());         System.out.println("无因咖啡,加一份牛奶描述:" + order2.getDescription());     } } 

结果打印:

order1费用:5.0 order1描述:longBlack order1加一份牛奶费用:7.0 order1加一份牛奶描述:牛奶 2.0 && longBlack order1加一份牛奶,加一份巧克力费用:10.0 order1加一份牛奶,加一份巧克力描述:巧克力 3.0 && 牛奶 2.0 && longBlack order1加一份牛奶,加两份巧克力费用:13.0 order1加一份牛奶,加两份巧克力描述:巧克力 3.0 && 巧克力 3.0 && 牛奶 2.0 && longBlack ============================== 无因咖啡费用:1.0 无因咖啡描述:无因咖啡 无因咖啡,加一份牛奶费用:3.0 无因咖啡,加一份牛奶描述:牛奶 2.0 && 无因咖啡 

3.小结

  • 装饰者模式可以带来比继承更加灵活性的扩展功能,使用更加方便,可以通过组合不同的装饰者对象来获取具有不同行为状态的多样化的结果。装饰者模式比继承更具良好的扩展性,完美的遵循开闭原则,继承是静态的附加责任,装饰者则是动态的附加责任。
  • 装饰类和被装饰类可以独立发展,不会相互耦合,装饰模式是继承的一个替代模式,装饰模式可以动态扩展一个实现类的功能。

相关内容

热门资讯

实测分享Wepoke工具软件透... 您好,这款游戏可以开挂的,确实是有挂的,需要了解加微【485275054】很多玩家在这款游戏中打牌都...
软件辅助挂Wepoke德州版本... 自定义新版系统规律,只需要输入自己想要的开挂功能,一键便可以生成出专用辅助器,不管你是想分享给你好友...
代打辅助挂微扑克代打原来是真的... 您好,微扑克这款游戏可以开挂的,确实是有挂的,需要了解加微【439369440】很多玩家在这款游戏中...
重大消息Wepoke实锤软件透... 您好,这款游戏可以开挂的,确实是有挂的,需要了解加微【757446909】很多玩家在这款游戏中打牌都...
总算了解Wepoke科技软件透... 总算了解Wepoke科技软件透明挂!太坏了其实确实真的是有挂的(有挂头条)(哔哩哔哩);软件透明挂更...
关于微扑克软件原来是真的有挂,... 关于微扑克软件原来是真的有挂,太夸张了原来真的有挂,详细教程(有挂详情);最新版2024是一款经典耐...
如何构建基于Python的推荐... 👽发现宝藏前些天发现了一个巨牛的人工智能学习网站,通俗易懂࿰...
【Python基础】一文搞懂:... 文章目录1 引言2 CSV 文件简介3 Python 中的 csv 模块4 写入 CSV 文件4.1...
Conda虚拟环境下libp1... Conda虚拟环境下libp11-kit.so.0: undefined symbol: ffi_t...
Python魔法之旅-魔法方法... 目录一、概述1、定义2、作用二、应用场景1、构造和析构2、操作符重载3、字符串和表示4、容器管理5、...