原文: https://howtodoinjava.com/design-patterns/structural/decorator-design-pattern/

在软件工程中,装饰器设计模式用于向类的特定实例添加其他功能或行为,而不会修改同一类的其他实例。 装饰器提供了子类别的灵活替代方案,以扩展功能。 请注意,以上描述暗示装饰对象会更改其行为,但不会更改其接口。

理解这种模式非常重要,因为一旦您了解了装饰技术,就可以赋予您(或其他人)对象新的职责,而无需对基础类进行任何代码更改。 有趣,不是吗? 这种模式也非常有用,并且经常会遇到 Java 面试问题关于设计模式的问题。

在这篇文章中,我们将讨论以下几点:

  • 设计参与者
  • 问题陈述
  • 使用装饰器设计模式的建议解决方案
  • 常见面试问题
  • 一些实际用法

设计参与者

装饰器模式的典型示意图如下所示。

Java 中的装饰器设计模式 - 图1

装饰设计模式参与者

以下是“装饰器设计”模式的参与者:

  • Component – 这是包装器,在运行时可以具有与其相关的其他职责。
  • ConcreteComponent – 是在程序中添加了其他职责的原始对象。
  • Decorator - 这是一个抽象类,其中包含对组件对象的引用,并且还实现了组件接口。
  • ConcreteDecorator - 它们扩展了装饰器,并在Component类的顶部构建了附加功能。

如果您仔细阅读这两行内容,那么您将了解它的工作方式如下:

您有一个实例,并将另一个实例放入其中。 它们都支持相同(或相似)的接口。 外面的一个是“装饰器”。 您在外面用一个。 它可以掩盖,更改或通过其内部实例的方法。

我总是喜欢以身作则。 因此,让我们解决一个问题。

问题陈述

让我们有一个常见的用例,我们必须将所有用户创建的报告显示给管理员。 现在,这些报告可以属于这些类别。

  • 客户报告
  • 支持报告

这两个报告都必须具有第一列作为到原始报告的链接,并且它们应具有不同的颜色。 可能应在不同弹出窗口大小中打开它们。 这些只是很少的事情,现实中可以实现很多。

一种可能的方法是扩展Report对象,然后具有两个单独的类ClientReportSupportReport。 在这两种方法中,定义私有方法,例如makeAnchorLink()designPopup()applyColor()等。 现在,在第一列数据单元格的某些获取器方法中调用这些方法。

Java 中的装饰器设计模式 - 图2

如果您在此处停止并且不再修改系统,则它将起作用。 但是,假设几天后,您被告知为两种报告应用不同的背景色。 现在,您只有一种方法可以在两个类中定义方法colorBackground(),并进行适当的调用。 将来系统中有更多报告时,问题将变得更加严重。

上面的解决方案明显违反了 SMART 类别设计中提到的开闭原则

使用装饰器设计模式的建议解决方案

上面的问题是装饰模式的理想选择。 请记住,当我们有一个需要扩展但设计不当的对象时,请进行装饰。

通过下面的类图可以轻松解决以上问题。 在这里,我仅用于支持报告。 也可以为客户报告构建类似的类层次结构。

Java 中的装饰器设计模式 - 图3

装饰器模式解决方案

如果我们已经实现了这样的解决方案,那么将来任何时候我们都可以添加其他修饰,而无需修改现有的类层次结构。 那是我们开始这篇文章的最终目标,对吧?

源代码列表

让我们看一看类的源代码,以了解其实际外观。

Report.java

  1. public interface Report {
  2. public Object[][] getReportData(final String reportId);
  3. public String getFirstColumnData();
  4. }

SupportReport.java

  1. public class SupportReport implements Report {
  2. @Override
  3. public Object[][] getReportData(String reportId) {
  4. return null;
  5. }
  6. @Override
  7. public String getFirstColumnData() {
  8. return "Support data";
  9. }
  10. }

ColumDecorator.java

  1. public abstract class ColumDecorator implements Report
  2. {
  3. private Report decoratedReport;
  4. public ColumDecorator(Report report){
  5. this.decoratedReport = report;
  6. }
  7. public String getFirstColumnData() {
  8. return decoratedReport.getFirstColumnData();
  9. }
  10. @Override
  11. public Object[][] getReportData(String reportId) {
  12. return decoratedReport.getReportData(reportId);
  13. }
  14. }

SupportLinkDecorator.java

  1. public class SupportLinkDecorator extends ColumDecorator{
  2. public SupportLinkDecorator(Report report) {
  3. super(report);
  4. }
  5. public String getFirstColumnData() {
  6. return addMoreInfo (super.getFirstColumnData()) ;
  7. }
  8. private String addMoreInfo(String data){
  9. return data + " - support link - ";
  10. }
  11. }

SupportPopupDecorator.java

  1. public class SupportPopupDecorator extends ColumDecorator{
  2. public SupportPopupDecorator(Report report) {
  3. super(report);
  4. }
  5. public String getFirstColumnData() {
  6. return addPopup (super.getFirstColumnData()) ;
  7. }
  8. private String addPopup(String data){
  9. return data + " - support popup - ";
  10. }
  11. }

DecoratorPatternTest.java

  1. public class DecoratorPatternTest {
  2. public static void main(String[] args) {
  3. //ClientPopupDecorator popupDecoratored = new ClientPopupDecorator(new ClientLinkDecorator(new ClientReport()));
  4. //System.out.println(popupDecoratored.getFirstColumnData());
  5. SupportPopupDecorator supportPopupDecoratored = new SupportPopupDecorator(new SupportLinkDecorator(new SupportReport()));
  6. System.out.println(supportPopupDecoratored.getFirstColumnData());
  7. }
  8. }
  9. Output:
  10. Support data - support link - - support popup -

要下载完整的源代码和 UML 图,请遵循文章结尾处的下载链接。

关于装饰模式的面试问题

A)如何确定何时使用装饰器模式?

如果我们深入了解该概念,则会发现装饰器设计模式具有多个需求指标以表明这是潜在的解决方案,例如:

  • 我们有一个需要扩展的对象。 例如一个需要其他“可选”功能(例如滚动条,标题栏和状态栏)的窗口控件。
  • 几个通过“装饰”支持扩展的对象。 通常,这些对象共享一个公共接口,特征或超类,有时还具有其他中间超类。
  • 装饰的对象(类或原型实例化)和装饰器对象具有一个或几个共同的特征。 为了确保该功能,装饰器&的装饰器具有公共接口,特征或类继承。

B)装饰器模式和适配器模式之间的区别

不可以。适配器模式用于将对象的接口转换为其他内容。 装饰器模式用于扩展对象的功能,同时保持其接口。 由于它们都“包裹”了一个对象,因此两者有时有时也被称为“包装模式”。

C)装饰器模式与子类别之间的差异

装饰器模式和子类之间的区别在于,在子类中,您可以“使用单个类”修饰实现接口的任何类。 假设我想给自己一个java.util.Map,每当我添加或删除键时,它都会打印一条消息。 如果我只实际使用过java.util.HashMap,那么我可以创建PrintingMap吗? 作为HashMap的子类并覆盖&删除。 但是,如果要创建TreeMap的打印版本,则可以创建PrintingTreeMap? (哪些代码与PrintingMap具有几乎相同的代码?

装饰器模式的常用用法:

1)Java IO 库类,例如BufferedInputStream bs = new BufferedInputStream(new FileInputStream(new File("File1.txt")));

2)在显示标签 jsp 库中的装饰器列中,例如:

  1. <display:table name="reportsViewResultTable" class="demoClass" id="reportsQueryViewResultTable">
  2. <display:column title="Report Id" sortable="true" property="reportDisplayId" decorator="com.comp.FirstColumnDataDecorator"></display:column>
  3. </display:table>

3)在 sitemesh 中使用装饰器,以提供一致的 UI 体验。

源码下载

祝您学习愉快!