首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >装饰器模式(Decorator)

装饰器模式(Decorator)

作者头像
qubianzhong
发布2019-07-01 14:03:31
发布2019-07-01 14:03:31
50000
代码可运行
举报
文章被收录于专栏:行者常至行者常至
运行总次数:0
代码可运行

装饰器模式(Decorator)

对客户透明的方式动态地给一个对象附加上更多的责任,同时又不改变其结构。装饰模式可以在不使用创造更多子类的情况下,将对象的功能加以扩展。
类图:
  • 1.抽象构件(Component)角色:给出一个抽象接口,以规范准备接收附加责任的对象。
  • 2.具体构件(ConcreteComponent)角色:定义一个将要接收附加责任的类。
  • 3.装饰(Decorator)角色:持有一个构件(Component)对象的实例,并定义一个与抽象构件接口一致的接口。
  • 4.具体装饰(ConcreteDecorator)角色:负责给构件对象“贴上”附加的责任。

例子

Java IO中就是典型的装饰器
代码语言:javascript
代码运行次数:0
运行
复制
//InputStream提供的基本方法(Component)
public abstract class InputStream implements Closeable {

}
代码语言:javascript
代码运行次数:0
运行
复制
//默认目标实现类(ConcreteComponent)
public class FileInputStream extends InputStream {

}
代码语言:javascript
代码运行次数:0
运行
复制
/*装饰实现类(FilterInputStream)一定是继承或实现原始接口(InputStream)的,内部有包含一个原始接口的超类(其实就是某个默认目标实现类)*/
//Decorator
public class FilterInputStream extends InputStream {
    /**
     * The input stream to be filtered.
     */
    protected volatile InputStream in;

    protected FilterInputStream(InputStream in) {
        this.in = in;
    }
}
代码语言:javascript
代码运行次数:0
运行
复制
//具体装饰类(ConcreteDecorator)
public class BufferedInputStream extends FilterInputStream {

    public BufferedInputStream(InputStream in) {
        this(in, DEFAULT_BUFFER_SIZE);
    }
}
代码语言:javascript
代码运行次数:0
运行
复制
//具体装饰类(ConcreteDecorator)
public class DataInputStream extends FilterInputStream implements DataInput {

    public DataInputStream(InputStream in) {
        super(in);
    }
}

总结

装饰器模式优点:
  • 1.装饰类和被装饰类可以独立发展,不会相互耦合。
  • 2.装饰模式是继承的一个替代模式,装饰模式可以动态扩展一个实现类的功能。就增加功能来说,装饰器模式相比生成子类更为灵活。
适用场景:
  • 1.扩展一个类的功能。
  • 2.动态增加功能,动态撤销。

Test

代码语言:javascript
代码运行次数:0
运行
复制
package com.qbz.test;

interface ICoffee {
    void showCoffee();

    float showPrice();
}

// 原始咖啡
class Coffee implements ICoffee {
    private String name;
    private float price;

    public Coffee(String name, float price) {
        this.name = name;
        this.price = price;
    }

    @Override
    public void showCoffee() {
        System.out.println(name + " coffee");
    }

    @Override
    public float showPrice() {
        return price;
    }
}

// 抽象装饰器
abstract class Decorator implements ICoffee {
    private ICoffee coffee;

    public void setCoffee(ICoffee coffee) {
        this.coffee = coffee;
    }

    @Override
    public void showCoffee() {
        coffee.showCoffee();
    }

    @Override
    public float showPrice() {
        return coffee.showPrice();
    }
}

// 加糖咖啡
class Sugar extends Decorator {
    @Override
    public void showCoffee() {
        System.out.println("加糖");
        super.showCoffee();
    }

    @Override
    public float showPrice() {
        return super.showPrice() + 5;
    }
}

// 加牛奶的咖啡
class SugarMilk extends Decorator {
    @Override
    public void showCoffee() {
        System.out.println("加糖、加牛奶");
        super.showCoffee();
    }

    @Override
    public float showPrice() {
        return super.showPrice() + 10;
    }
}

public class Main {

    public static void main(String[] args) {
        Coffee coffee = new Coffee("拿铁", 20);
        // 加糖
        Decorator sugar = new Sugar();
        sugar.setCoffee(coffee);
        sugar.showCoffee();
        System.out.println(sugar.showPrice());
        // 加糖,加牛奶的咖啡
        Decorator sugarmilk = new SugarMilk();
        sugarmilk.setCoffee(coffee);
        sugarmilk.showCoffee();
        System.out.println(sugarmilk.showPrice());
    }

}
输出
代码语言:javascript
代码运行次数:0
运行
复制
加糖
拿铁 coffee
25.0
加糖、加牛奶
拿铁 coffee
30.0
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2016年08月30日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 装饰器模式(Decorator)
    • 对客户透明的方式动态地给一个对象附加上更多的责任,同时又不改变其结构。装饰模式可以在不使用创造更多子类的情况下,将对象的功能加以扩展。
    • 类图:
  • 例子
    • Java IO中就是典型的装饰器
  • 总结
    • 装饰器模式优点:
    • 适用场景:
  • Test
    • 输出
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档