装饰模式

概述

23种设计模式之一,英文叫DecoratorPattern,中文也叫装饰模式、修饰模式。装饰模式是在不改变类文件和不使用继承的情况下,运行期动态扩展一个对象的功能。原理是:增加一个修饰类包裹原来的类,包裹的方式一般是通过在将原来的对象作为修饰类的构造函数的参数。装饰类实现新的功能,但是,在不需要用到新功能的地方,它可以直接调用原来的类中的方法。修饰类必须和原来的类有相同的接口(没有接口可以直接继承自原来的类)。修饰模式是类继承的另外一种选择。类继承在编译时候增加行为,而装饰模式是在运行时增加行为。

UML

实现

Component.java,接口。

public interface Component {   void operation();}

ConcreteComponent.java,原有类实现(需要扩展)。

public class ConcreteComponent implements Component {   public void operation() {      System.out.println("具体对象的操作");   }}

Decorator.java,抽象修饰类。

public abstract class Decorator implements Component {   protected Component component;   public Decorator(Component component) {      this.component = component;   }   public void operation() {      if (component !=null)        component.operation();   }}

ConcreteDecoratorA.java,实际修饰类A。

public class ConcreteDecoratorA extends Decorator {   public ConcreteDecoratorA(Component component) {      super(component);   }   @Override   public void operation() {      super.operation();      System.out.println("对象A扩展的操作");   }}

ConcreteDecoratorB.java,实际修饰类B。

public class ConcreteDecoratorB extends Decorator {   public ConcreteDecoratorB(Component component) {      super(component);   }   @Override   public void operation() {      super.operation();      System.out.println("对象B扩展的操作");   }}

DecoratorTest.java,客户端。

public class DecoratorTest {   public static void main(String[] args) {      Component component =         new ConcreteDecoratorB(        new ConcreteDecoratorA(        new ConcreteComponent()));      component.operation();   }}

输出结果:

具体对象的操作对象A扩展的操作对象B扩展的操作

上面客户端的调用方式是不是和如下的代码有些类似,没错,Java中的I/O类库使用的就是装饰模式。

BufferedInputStream bis = new BufferedInputStream(new FileInputStream(""));

上面就是装饰模式的模型,如果有不明白,可以结合代码、UML、定义一起看一下。

实例

需求

一个窗口系统中的窗口,允许这个窗口内容滚动,我们希望给它添加水平或垂直滚动条(维基百科)。

实现

Window.java,窗口接口。

public interface Window {   public void draw();   public String getDescription();}

SimpleWindow.java,简单窗口,不带任何修饰。

public class SimpleWindow implements Window {   public void draw() {  // draw window   }   public String getDescription() {      return "simple window";   }}

WindowDecorator.java,窗口装饰抽象类。

public abstract class WindowDecoratorimplements Window {   protected Window decoratedWindow;// the Window being decorated   public WindowDecorator(Window decoratedWindow) {      this.decoratedWindow = decoratedWindow;   }}

HorizontalScrollBarDecorator.java,横向滚动条装饰类。

public class HorizontalScrollBarDecorator extends WindowDecorator {   public HorizontalScrollBarDecorator(Window decoratedWindow) {      super(decoratedWindow);   }   public void draw() {      drawHorizontalScrollBar();      decoratedWindow.draw();   }   private void drawHorizontalScrollBar() {      // draw the horizontal scrollbar   }   public String getDescription() {      return decoratedWindow.getDescription() + ", including horizontal scrollbars";   }}

VerticalScrollBarDecorator.java,纵向滚动条装饰类。

public class VerticalScrollBarDecorator extends WindowDecorator {   public VerticalScrollBarDecorator(Window decoratedWindow) {      super(decoratedWindow);   }   public void draw() {      drawVerticalScrollBar();      decoratedWindow.draw();   }   private void drawVerticalScrollBar() {      // draw the vertical scrollbar   }   public String getDescription() {      return decoratedWindow.getDescription() + ", including vertical scrollbars";   }}

DecoratedWindowTest.java,客户端。

public class DecoratedWindowTest {   public static void main(String[] args) {      // create a decorated Window with horizontal and vertical scrollbars      Window decoratedWindow =         new HorizontalScrollBarDecorator(        new VerticalScrollBarDecorator(        new SimpleWindow()));      // print the Window's description      System.out.println(decoratedWindow.getDescription());   }}

输出结果:

simple window, including vertical scrollbars, including horizontal scrollbars

将不同的装饰区分开来,并且和原有的窗口分开,这样通过包装,我可以创建一个只带横向(纵向)滚动条的窗口,也可以创建不带滚动条的窗口,任意组合。

总结

装饰模式是不使用继承的情况下,可以动态扩展一个类,并且比继承更灵活(上面的实例)。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 工厂方法模式

    概述        工厂方法模式,英文Factory method pattern,工厂方法模式是简单工厂模式的进化版, 看本文之间最好先看一下简单工厂模式。工...

    高爽
  • 策略模式

    概述 策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。 UML ? 实现 Strate...

    高爽
  • Java线程(三):线程协作-生产者/消费者问题

            上一篇讲述了线程的互斥(同步),但是在很多情况下,仅仅同步是不够的,还需要线程与线程协作(通信),生产者/消费者问题是一个经典的线程同步以及通信...

    高爽
  • 23种设计模式详解(六)

    南风
  • (52) 抽象容器类 / 计算机程序的思维逻辑

    查看历史文章,请点击上方链接关注公众号。 从38节到51节,我们介绍的都是具体的容器类,上节我们提到,所有具体容器类其实都不是从头构建的,它们都继承了一些抽象容...

    swiftma
  • Java设计模式总汇二(小白也要飞)

    PS:上一篇我介绍了适配器设计模式、单例设计模式、静态代理设计模式、简单工厂设计模式,如果没有看过第一篇的小火鸡可以点这个看看http://www.cnblog...

    cMusketeer
  • 代理模式

    mySoul
  • 23种设计模式详解(五)

    南风
  • Head First设计模式——蝇量模式和解释器模式

    在一个设计房子的平台中,周围要加上一些树,树有一个坐标XY坐标位置,而且可以根据树的年龄动态将自己绘制出来。如果我们创建许多树之后,会有许多树的实例对象。使用一...

    SpringSun
  • 使用angular4和asp.net core 2 web api做个练习项目(一)

    这是一篇学习笔记. angular 5 正式版都快出了, 不过主要是性能升级. 我认为angular 4还是很适合企业的, 就像.net一样. 我用的是wind...

    solenovex

扫码关注云+社区

领取腾讯云代金券