【设计模式】用Java实现装饰模式

简介: 装饰模式(Decorator Pattern)是一种结构型设计模式,它允许在不改变现有对象结构的情况下,动态地将责任附加到对象上。装饰模式通过将对象包装在装饰器对象中,形成一条装饰链,每个装饰器对象都可以在被装饰对象的行为前后加上自己的行为。

一.装饰模式介绍与使用场景


装饰模式(Decorator Pattern)是一种结构型设计模式,它允许在不改变现有对象结构的情况下,动态地将责任附加到对象上。装饰模式通过将对象包装在装饰器对象中,形成一条装饰链,每个装饰器对象都可以在被装饰对象的行为前后加上自己的行为。


装饰模式的核心思想是通过组合而不是继承来扩展对象的功能。它遵循开放-关闭原则,即对扩展开放,对修改关闭,使得我们可以在不修改现有代码的情况下,灵活地添加新的功能。


应用场景:


1.需要在不改变现有对象结构的情况下,动态地为对象添加额外的功能。装饰模式通过组合而不是继承来实现功能的扩展,使得可以在运行时动态地添加、删除或修改对象的行为。

2.需要对对象的功能进行多层次的包装和组合。装饰模式可以形成一条装饰链,每个装饰器可以独立地对被装饰对象进行包装,从而实现多个装饰器的叠加效果,灵活地组合功能。

3.需要动态地为对象添加或移除功能。由于装饰模式是基于对象组合的,因此可以在运行时动态地添加或移除装饰器对象,实现对对象功能的动态调整。

4.需要对一些特定对象或对象集合进行特殊处理,而不影响其他对象。装饰模式允许针对特定对象或对象集合创建特定的装饰器,对它们进行个性化的功能扩展,而不会影响其他对象。


装饰模式在许多应用中都有广泛的应用,例如:

Java I/O流中的输入流和输出流的装饰器类(如BufferedInputStream和BufferedOutputStream)。


图形界面中的组件装饰,如对按钮添加边框、背景色等装饰效果。


日志记录器中的功能扩展,如在日志中添加时间戳、记录日志级别等功能。


总之,装饰模式适用于需要动态地为对象添加额外功能、实现多层次包装和组合、动态调整对象功能以及对特定对象进行特殊处理的场景。它提供了一种灵活、可扩展的方式来扩展对象的功能,同时遵循开放-关闭原则,使得系统的设计更加灵活和可维护。


二.装饰模式实现


下面是一个使用Java实现装饰模式的简单示例:


首先,定义一个抽象基础组件 Component,它声明了一个操作方法:

interface Component {
    void operation();
}

然后,创建具体的基础组件类 ConcreteComponent,它实现了抽象组件接口:

class ConcreteComponent implements Component {
    public void operation() {
        System.out.println("Performing operation in ConcreteComponent.");
    }
}

接下来,定义装饰器抽象类 Decorator,它也实现了抽象组件接口,并维持一个指向抽象组件的引用:

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

然后,创建具体的装饰器类 ConcreteDecorator,它继承自装饰器抽象类,并在装饰方法中添加额外的功能:

class ConcreteDecorator extends Decorator {
    public ConcreteDecorator(Component component) {
        super(component);
    }
    public void operation() {
        // 在原有功能前后添加额外的功能
        System.out.println("Before operation in ConcreteDecorator.");
        super.operation();
        System.out.println("After operation in ConcreteDecorator.");
    }
}

最后,我们可以在客户端中使用这些类来创建和使用装饰对象:

public class Client {
    public static void main(String[] args) {
        // 创建基础组件对象
        Component component = new ConcreteComponent();
        // 使用装饰器包装基础组件对象
        Component decoratedComponent = new ConcreteDecorator(component);
        // 调用装饰后的操作方法
        decoratedComponent.operation();
    }
}

输出结果为:

Before operation in ConcreteDecorator.
Performing operation in ConcreteComponent.
After operation in ConcreteDecorator.

通过装饰模式,我们可以在不改变基础组件类的情况下,动态地为对象添加额外的功能。装饰器类和基础组件类之间使用组合关系,形成一条装饰链,每个装饰器都可以在调用基础组件的操作前后添加自己的行为。这样,我们可以灵活地扩展对象的功能,而无需修改现有代码。


下面再举一个例子,在实际项目中模拟咖啡店的咖啡订单系统。


假设我们有一个基础咖啡类 Coffee,它实现了咖啡接口:

interface Coffee {
    String getDescription();
    double getCost();
}

然后,创建具体的咖啡类 EspressoCappuccino,它们实现了咖啡接口并提供具体的描述和价格:

class Espresso implements Coffee {
    public String getDescription() {
        return "Espresso";
    }
    public double getCost() {
        return 2.0;
    }
}
class Cappuccino implements Coffee {
    public String getDescription() {
        return "Cappuccino";
    }
    public double getCost() {
        return 3.5;
    }
}

接下来,定义装饰器抽象类 CoffeeDecorator,它也实现了咖啡接口,并维持一个指向咖啡对象的引用:

abstract class CoffeeDecorator implements Coffee {
    protected Coffee coffee;
    public CoffeeDecorator(Coffee coffee) {
        this.coffee = coffee;
    }
    public String getDescription() {
        return coffee.getDescription();
    }
    public double getCost() {
        return coffee.getCost();
    }
}

然后,创建具体的装饰器类 MilkDecoratorSugarDecorator,它们继承自装饰器抽象类,并在装饰方法中添加额外的功能:

class MilkDecorator extends CoffeeDecorator {
    public MilkDecorator(Coffee coffee) {
        super(coffee);
    }
    public String getDescription() {
        return super.getDescription() + ", Milk";
    }
    public double getCost() {
        return super.getCost() + 0.5;
    }
}
class SugarDecorator extends CoffeeDecorator {
    public SugarDecorator(Coffee coffee) {
        super(coffee);
    }
    public String getDescription() {
        return super.getDescription() + ", Sugar";
    }
    public double getCost() {
        return super.getCost() + 0.2;
    }
}

最后,我们可以在客户端中使用这些类来创建和使用装饰对象:

public class Client {
    public static void main(String[] args) {
        // 创建基础咖啡对象
        Coffee espresso = new Espresso();
        Coffee cappuccino = new Cappuccino();
        // 使用装饰器包装咖啡对象
        Coffee milkEspresso = new MilkDecorator(espresso);
        Coffee sugarCappuccino = new SugarDecorator(cappuccino);
        // 输出咖啡描述和价格
        System.out.println(milkEspresso.getDescription() + " - $" + milkEspresso.getCost());
        System.out.println(sugarCappuccino.getDescription() + " - $" + sugarCappuccino.getCost());
    }
}

输出结果为:

Espresso, Milk - $2.5
Cappuccino, Sugar - $3.7

通过装饰模式,我们可以在不修改基础咖啡类的情况下,动态地为咖啡对象添加额外的功能,如添加牛奶和糖等。装饰器类和咖啡类之间使用组合关系,形成一条装饰链,每个装饰器都可以在调用咖啡对象的方法前后添加自己的行为。这样,我们可以根据客户的需求灵活地组合和定制咖啡,而不需要创建大量的子类。

相关文章
|
6天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
|
2天前
|
设计模式 Java
Java设计模式:组合模式的介绍及代码演示
组合模式是一种结构型设计模式,用于将多个对象组织成树形结构,并统一处理所有对象。例如,统计公司总人数时,可先统计各部门人数再求和。该模式包括一个通用接口、表示节点的类及其实现类。通过树形结构和节点的通用方法,组合模式使程序更易扩展和维护。
Java设计模式:组合模式的介绍及代码演示
|
6天前
|
设计模式 安全 算法
【Java面试题汇总】设计模式篇(2023版)
谈谈你对设计模式的理解、七大原则、单例模式、工厂模式、代理模式、模板模式、观察者模式、JDK中用到的设计模式、Spring中用到的设计模式
【Java面试题汇总】设计模式篇(2023版)
|
6天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
21天前
|
设计模式 缓存 算法
揭秘策略模式:如何用Java设计模式轻松切换算法?
【8月更文挑战第30天】设计模式是解决软件开发中特定问题的可重用方案。其中,策略模式是一种常用的行为型模式,允许在运行时选择算法行为。它通过定义一系列可互换的算法来封装具体的实现,使算法的变化与客户端分离。例如,在电商系统中,可以通过定义 `DiscountStrategy` 接口和多种折扣策略类(如 `FidelityDiscount`、`BulkDiscount` 和 `NoDiscount`),在运行时动态切换不同的折扣逻辑。这样,`ShoppingCart` 类无需关心具体折扣计算细节,只需设置不同的策略即可实现灵活的价格计算,符合开闭原则并提高代码的可维护性和扩展性。
37 2
|
21天前
|
设计模式 Java
Java 设计模式之谜:工厂模式与抽象工厂模式究竟隐藏着怎样的神奇力量?
【8月更文挑战第30天】在Java编程中,设计模式为常见问题提供了高效解决方案。工厂模式与抽象工厂模式是常用的对象创建型设计模式,能显著提升代码的灵活性、可维护性和可扩展性。工厂模式通过定义创建对象的接口让子类决定实例化哪个类;而抽象工厂模式则进一步提供了一个创建一系列相关或相互依赖对象的接口,无需指定具体类。这种方式使得系统更易于扩展和维护。
30 1
|
21天前
|
设计模式 Java
重构你的代码:探索Java中的混合、装饰器与组合设计模式
【8月更文挑战第30天】在软件开发中,设计模式为特定问题提供了结构化的解决方案,使代码更易理解、维护及扩展。本文将介绍三种常用的 Java 设计模式:混合模式、装饰器模式与组合模式,并附有示例代码展示实际应用。混合模式允许通过继承多个接口或抽象类实现多重继承;装饰器模式可在不改变对象结构的情况下动态添加新功能;组合模式则通过树形结构表示部分-整体层次,确保客户端处理单个对象与组合对象时具有一致性。
15 1
|
3月前
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
42 4
|
1月前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
4月前
|
设计模式 安全 Java
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式