2019-06-15 10:50  阅读(2346)
文章分类:史上最全设计模式 文章标签:设计模式Java 设计模式
©  原文作者:Liuwei-Sunny 原文地址:https://blog.csdn.net/lovelion

作者:Liuwei-Sunny

出处:https://blog.csdn.net/lovelion


12.3 完整解决方案

为了让系统具有更好的灵活性和可扩展性,克服继承复用所带来的问题,Sunny公司开发人员使用装饰模式来重构图形界面构件库的设计,其中部分类的基本结构如图12-4所示:

2019061510052_1.png

图12-4 图形界面构件库结构图

在图12-4中,Component充当抽象构件类,其子类Window、TextBox、ListBox充当具体构件类,Component类的另一个子类ComponentDecorator充当抽象装饰类,ComponentDecorator的子类ScrollBarDecorator和BlackBorderDecorator充当具体装饰类。完整代码如下所示:

//抽象界面构件类:抽象构件类,为了突出与模式相关的核心代码,对原有控件代码进行了大量的简化abstractclassComponent{      public abstractvoiddisplay();} //窗体类:具体构件类classWindowextendsComponent{      public voiddisplay()      {             System.out.println("显示窗体!");      }} //文本框类:具体构件类classTextBoxextendsComponent{      public voiddisplay()      {             System.out.println("显示文本框!");      }} //列表框类:具体构件类classListBoxextendsComponent{      public voiddisplay()      {             System.out.println("显示列表框!");      }} //构件装饰类:抽象装饰类classComponentDecoratorextendsComponent{      privateComponentcomponent; //维持对抽象构件类型对象的引用       publicComponentDecorator(Component component) //注入抽象构件类型的对象      {             this.component=component;      }       publicvoiddisplay()      {             component.display();      }} //滚动条装饰类:具体装饰类classScrollBarDecoratorextends ComponentDecorator{      publicScrollBarDecorator(Component component)      {             super(component);      }       publicvoiddisplay()      {             this.setScrollBar();             super.display();      }       public voidsetScrollBar()      {             System.out.println("为构件增加滚动条!");      }} //黑色边框装饰类:具体装饰类classBlackBorderDecoratorextends ComponentDecorator{      publicBlackBorderDecorator(Component component)      {             super(component);      }       publicvoiddisplay()      {             this.setBlackBorder();             super.display();      }       public voidsetBlackBorder()      {             System.out.println("为构件增加黑色边框!");      }}
//抽象界面构件类:抽象构件类,为了突出与模式相关的核心代码,对原有控件代码进行了大量的简化abstractclassComponent{      public abstractvoiddisplay();} //窗体类:具体构件类classWindowextendsComponent{      public voiddisplay()      {             System.out.println("显示窗体!");      }} //文本框类:具体构件类classTextBoxextendsComponent{      public voiddisplay()      {             System.out.println("显示文本框!");      }} //列表框类:具体构件类classListBoxextendsComponent{      public voiddisplay()      {             System.out.println("显示列表框!");      }} //构件装饰类:抽象装饰类classComponentDecoratorextendsComponent{      privateComponentcomponent; //维持对抽象构件类型对象的引用       publicComponentDecorator(Component component) //注入抽象构件类型的对象      {             this.component=component;      }       publicvoiddisplay()      {             component.display();      }} //滚动条装饰类:具体装饰类classScrollBarDecoratorextends ComponentDecorator{      publicScrollBarDecorator(Component component)      {             super(component);      }       publicvoiddisplay()      {             this.setScrollBar();             super.display();      }       public voidsetScrollBar()      {             System.out.println("为构件增加滚动条!");      }} //黑色边框装饰类:具体装饰类classBlackBorderDecoratorextends ComponentDecorator{      publicBlackBorderDecorator(Component component)      {             super(component);      }       publicvoiddisplay()      {             this.setBlackBorder();             super.display();      }       public voidsetBlackBorder()      {             System.out.println("为构件增加黑色边框!");      }}

编写如下客户端测试代码:

classClient{      public staticvoidmain(Stringargs[])      {             Componentcomponent,componentSB; //使用抽象构件定义             component=newWindow();//定义具体构件             componentSB=new ScrollBarDecorator(component);//定义装饰后的构件             componentSB.display();      }}
classClient{      public staticvoidmain(Stringargs[])      {             Componentcomponent,componentSB; //使用抽象构件定义             component=newWindow();//定义具体构件             componentSB=new ScrollBarDecorator(component);//定义装饰后的构件             componentSB.display();      }}

编译并运行程序,输出结果如下:

为构件增加滚动条!显示窗体!
为构件增加滚动条!显示窗体!

在客户端代码中,我们先定义了一个Window类型的具体构件对象component,然后将component作为构造函数的参数注入到具体装饰类ScrollBarDecorator中,得到一个装饰之后对象componentSB,再调用componentSB的display()方法后将得到一个有滚动条的窗体。如果我们希望得到一个既有滚动条又有黑色边框的窗体,不需要对原有类库进行任何修改,只需将客户端代码修改为如下所示:

classClient{      public staticvoidmain(Stringargs[])      {             Component component,componentSB,componentBB;//全部使用抽象构件定义             component=newWindow();             componentSB=new ScrollBarDecorator(component);             componentBB=new BlackBorderDecorator(componentSB);//将装饰了一次之后的对象继续注入到另一个装饰类中,进行第二次装饰             componentBB.display();      }}
classClient{      public staticvoidmain(Stringargs[])      {             Component component,componentSB,componentBB;//全部使用抽象构件定义             component=newWindow();             componentSB=new ScrollBarDecorator(component);             componentBB=new BlackBorderDecorator(componentSB);//将装饰了一次之后的对象继续注入到另一个装饰类中,进行第二次装饰             componentBB.display();      }}

编译并运行程序,输出结果如下:

为构件增加黑色边框!为构件增加滚动条!显示窗体!
为构件增加黑色边框!为构件增加滚动条!显示窗体!

我们可以将装饰了一次之后的componentSB对象注入另一个装饰类BlackBorderDecorator中实现第二次装饰,得到一个经过两次装饰的对象componentBB,再调用componentBB的display()方法即可得到一个既有滚动条又有黑色边框的窗体。

如果需要在原有系统中增加一个新的具体构件类或者新的具体装饰类,无须修改现有类库代码,只需将它们分别作为抽象构件类或者抽象装饰类的子类即可。与图12-2所示的继承结构相比,使用装饰模式之后将大大减少了子类的个数,让系统扩展起来更加方便,而且更容易维护,是取代继承复用的有效方式之一。

点赞(0)
版权归原创作者所有,任何形式转载请联系作者; Java 技术驿站 >> 扩展系统功能——装饰模式(三):图形界面构件库的装饰模式解决方案
上一篇
扩展系统功能——装饰模式(二):装饰模式概述
下一篇
扩展系统功能——装饰模式(四):透明装饰模式与半透明装饰模式,装饰模式注意事项,装饰模式总结