zoukankan      html  css  js  c++  java
  • 装饰者模式

    • 定义与类型
      • 定义:在不改变原有对象的基础上,将功能附加到对象上
      • 提供了比继承更有弹性的替代方案(扩展原有对象功能)
      • 类型:结构型
    • 适用场景
      • 扩展一个类的功能或给一个类添加附加职责
      • 动态的给一个对象添加功能,这些功能可以再动态地撤销
    • 优点
      • 继承的有力补充,比继承灵活,不改变原有对象的情况下给一个对象扩展功能
      • 通过使用不同装饰类以及这些装饰类的排列组合,可以实现不同效果
      • 符合开闭原则
    • 缺点
      • 会出现更多的代码,更多的类,增加程序复杂性
      • 动态装饰时,多层装饰时会更负责
    • 相关设计模式
      • 装饰者模式和代理模式
      • 装饰者模式和适配器模式

    现在有一个场景,煎饼馃子的老板卖煎饼,煎饼可以加蛋或者香肠。这个类怎么实现呢?

    一种方式就是使用继承:我有一个煎饼馃子的类,要加蛋的话添加一个加蛋的类,要加香肠的话添加一个香肠的类。类图如下所示:

    继承的方式

    通过继承的方式来扩展有什么弊端呢?第一个缺点是容易导致「类爆炸」,就是我加什么都得创建一个新的类;第二个缺点是我想要加两个蛋怎么加,也得重新创建一个类。

    那怎么办呢?可以用装饰者模式

    装饰者模式

    • ABattercake 是一个抽象类,其中包括计算价格和添加描述的方法;

    • AbstractDecorator 也是一个抽象类,它就是一个抽象的装饰者,我们可以用一个构造器注入一个 ABattercake;

    • 1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      
      public abstract class AbstractDecorator extends ABattercake {
          private ABattercake aBattercake;
      
          public AbstractDecorator(ABattercake aBattercake) {
              this.aBattercake = aBattercake;
          }
      
          protected abstract void doSomething();
      
          @Override
          protected String getDesc() {
              return this.aBattercake.getDesc();
          }
      
          @Override
          protected int cost() {
              return this.aBattercake.cost();
          }
      }
      
    • Battercake 和 EggDecorator 是 AbstractDecorator 具体实现

    • 1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      
      public class EggDecorator extends AbstractDecorator {
          public EggDecorator(ABattercake aBattercake) {
              super(aBattercake);
          }
      
          @Override
          protected void doSomething() {
      
          }
      
          @Override
          protected String getDesc() {
              return super.getDesc()+" 加一个鸡蛋";
          }
      
          @Override
          protected int cost() {
              return super.cost()+1;
          }
      }
      

    这样我想要几个蛋就加创建几个蛋对象,想要几个香肠创建几个香肠对象。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    
    public class Test {
        public static void main(String[] args) {
            ABattercake aBattercake;
            aBattercake = new Battercake();
            aBattercake = new EggDecorator(aBattercake);
            aBattercake = new EggDecorator(aBattercake);
            aBattercake = new SausageDecorator(aBattercake);
    
            System.out.println(aBattercake.getDesc()+" 销售价格:"+aBattercake.cost());
        }
    }
    

    装饰者模式同样会产生「类爆炸」的问题,但是比继承的方式好一点。同时装饰者模式需要创建更多的对象,程序的复杂性也提高了。

  • 相关阅读:
    洛谷 P1464 Function【动态规划(递推)/记忆化搜索(递归)】
    洛谷 P1426 小鱼会有危险吗【模拟/题意理解】
    洛谷 P2089 烤鸡【DFS递归/10重枚举】
    洛谷 P1579 哥德巴赫猜想(升级版)【筛素数/技巧性枚举/易错】
    洛谷 P1618 三连击(升级版)【DFS/next_permutation()/技巧性枚举/sprintf】
    sprintf 心得
    洛谷 P1478 陶陶摘苹果(升级版)【贪心/结构体排序/可用01背包待补】
    吉首大学 问题 L: 小李子的老年生活
    AtCoder Beginner Contest 084 D
    一维差值维护心得
  • 原文地址:https://www.cnblogs.com/shuiyj/p/13185174.html
Copyright © 2011-2022 走看看