Java设计模式-结构型模式-装饰模式

装饰模式

  • 装饰模式
    • 角色
    • 案例
    • 装饰模式与静态代理的区别

装饰模式

允许向一个现有的对象动态地添加新的功能,同时不改变其结构。它是继承的一种替代方案,可以动态地扩展对象。有点像静态代理

角色

装饰者模式有四种角色 抽象被装饰者,被装饰者,装饰者和抽象被装饰者

  • 抽象被装饰者:一般是一个接口,包含需要被装饰的方法
  • 被装饰者:实现 抽象被装饰者 接口,代表被装饰的原始对象。
  • 抽象装饰者:继承或实现抽象被装饰者,内部含有一个 抽象被装饰者的属性,调用或增强 被装饰者的方法
  • 装饰者:实现抽象装饰者的方法,给被装饰对象增加具体的职责

案例

今天不开包子店,想开一家奶茶店。产品有普通奶茶,为了奶茶种类丰富,需要支持添加多种小料,椰果,芋圆等,且可以自由组合。
根据要求,先建立起基础类
接口 Tea,用来表示 抽象被装饰类,MilkTea 用来表示 被装饰类

/**
 * 抽象被装饰者
 **/
public interface Tea {
    String getName();

    int getPrice();
}
/**
 * 被装饰者:奶茶
 **/
public class MilkTea implements Tea{
    @Override
    public String getName() {
        return "奶茶";
    }

    @Override
    public int getPrice() {
        return 10;
    }
}

分析下问题,想要支持可以自由组合的方式来 增强奶茶类,普通的继承和组合 很难实现。所以才使用装饰者模式
下面是装饰者的代码

/**
 * 抽象装饰者:奶茶小料
 **/
public abstract class AbsDecoratorIngredients implements Tea{

    final private Tea tea;

    public AbsDecoratorIngredients(Tea tea) {
        this.tea = tea;
    }

    @Override
    public int getPrice() {
        //执行目标对象原本的行为
        return tea.getPrice();
    }

    @Override
    public String getName() {
        //执行目标对象原本的行为
        return tea.getName();
    }
}

/**
 * 装饰者:椰果
 **/
public class DecoratorCoconut extends AbsDecoratorIngredients{

    public DecoratorCoconut(Tea tea) {
        super(tea);
    }

    @Override
    public int getPrice() {
        /*被装饰者的价格 + 当前椰果的价格*/
        return super.getPrice() + 2;
    }

    @Override
    public String getName() {
        return super.getName() + " + " + "椰果";
    }
}
/**
 * 装饰者:芋圆
 **/
public class DecoratorTaroBall extends AbsDecoratorIngredients{

    public DecoratorTaroBall(Tea tea) {
        super(tea);
    }

    @Override
    public int getPrice() {
        /*被装饰者的价格 + 当前椰果的价格*/
        return super.getPrice() + 2;
    }

    @Override
    public String getName() {
        return super.getName() + " + " + "椰果";
    }
}

测试代码:

public class DecoratorTest {
    public static void main(String[] args) {
        /*给我一杯奶茶*/
        System.out.println("==========================给我一杯奶茶====================================");
        Tea milkTea = new MilkTea();
        System.out.println(milkTea.getName() + " = " + milkTea.getPrice());


        /*给我一杯珍珠奶茶*/
        System.out.println("===========================给我一杯珍珠奶茶===================================");
        milkTea = new DecoratorTaroBall(milkTea);
        System.out.println(milkTea.getName() + " = " + milkTea.getPrice());


        /*给我一杯珍珠椰果奶茶*/
        System.out.println("============================给我一杯珍珠椰果奶茶==================================");
        milkTea = new DecoratorCoconut(milkTea);
        System.out.println(milkTea.getName() + " = " + milkTea.getPrice());
    }
}

输出:
==========================给我一杯奶茶====================================
奶茶 = 10
===========================给我一杯珍珠奶茶===================================
奶茶 + 椰果 = 12
============================给我一杯珍珠椰果奶茶==================================
奶茶 + 椰果 + 椰果 = 14

看到这里,可能有人发现 抽象装饰者类 好像可以省略。
是的,这个例子是可以省略的,
这里只是写的 装饰者模式的标准写法。使用一个抽象装饰者 是为了 将 抽象被装饰者 和 装饰者 解耦。

一旦把 抽象装饰者省略,大家可能发现这个写法是不是很眼熟,是不是很像 静态代理。
其实 设计模式 就是这样,23种设计模式只是应对不同的场景,不同的设计模式总会有相似之处。
设计模式来源于设计原则,万变不离其宗,稍微变换下就是另一种设计模式。

下面看下 装饰和静态代理的区别

装饰模式与静态代理的区别

装饰模式:目标对象由外界传入,目的是为了增强该对象
静态代理:目标对象由内部生成,目的是隐藏和保护该对象
装饰模式一般会迭代传入不同的对象,一步一步的增强方法
静态代理一般只传入一个对象,只调用一层

你可能感兴趣的:(设计模式,java,设计模式)