java设计模式——装饰者模式

定义: 装饰者模式是一种结构型设计模式,它允许动态地给对象添加新的功能,而不会改变其原有的结构。与继承不同,装饰者模式通过组合而不是继承来扩展对象的功能,这样可以有效地避免类爆炸问题(多个子类的冗余)。

在装饰者模式中,通常有以下几个关键角色:

抽象组件:定义对象的接口,可以是接口或抽象类。具体组件和装饰者都实现或继承该组件。

具体组件:实现抽象组件接口的具体类,它是被装饰的对象。

装饰者:实现抽象组件接口的类,内部维护一个抽象组件的引用,用于对被装饰对象进行扩展。

具体装饰者:继承装饰者并扩展其功能,可以为被装饰对象动态添加新功能。

优点

灵活性:通过组合而不是继承来扩展对象的功能,可以在运行时选择不同的装饰者动态组合对象。

遵循开闭原则:装饰者模式允许对功能进行扩展,而无需修改现有的代码。

减少子类的冗余:避免类层次的复杂性和继承的弊端。

缺点

复杂性增加:使用装饰者模式会增加系统中类的数量和对象的层次,增加理解和调试的难度。

难以维护:多个装饰者叠加时,调试可能变得困难,因为可能需要跟踪多个装饰者的行为。

实现示例

假设我们有一个咖啡店系统,每种咖啡有不同的类型(如普通咖啡、加牛奶的咖啡、加糖的咖啡等)。我们想要通过装饰者模式来灵活地添加配料,而不是为每种组合创建不同的类。

  1. 定义抽象组件 Beverage
java 复制代码
// 抽象组件
public abstract class Beverage {
    // 每种饮料都有一个描述和一个价格
    protected String description = "Unknown Beverage";

    public String getDescription() {
        return description;
    }

    public abstract double cost();
}
  1. 定义具体组件 Coffee
java 复制代码
// 具体组件 - 咖啡
public class Coffee extends Beverage {

    public Coffee() {
        description = "Coffee";
    }

    @Override
    public double cost() {
        return 5.0; // 基本咖啡的价格
    }
}
  1. 定义装饰者 CondimentDecorator
java 复制代码
// 抽象装饰者 - 调料装饰器
public abstract class CondimentDecorator extends Beverage {
    // 强制要求具体装饰者必须实现 getDescription 方法
    public abstract String getDescription();
}
  1. 定义具体装饰者 MilkSugar
java 复制代码
// 具体装饰者 - 牛奶
public class Milk extends CondimentDecorator {
    // 被装饰的对象
    Beverage beverage;

    public Milk(Beverage beverage) {
        this.beverage = beverage;
    }

    @Override
    public String getDescription() {
        return beverage.getDescription() + ", Milk";
    }

    @Override
    public double cost() {
        return 1.5 + beverage.cost(); // 牛奶的价格加上原始饮料的价格
    }
}

// 具体装饰者 - 糖
public class Sugar extends CondimentDecorator {
    // 被装饰的对象
    Beverage beverage;

    public Sugar(Beverage beverage) {
        this.beverage = beverage;
    }

    @Override
    public String getDescription() {
        return beverage.getDescription() + ", Sugar";
    }

    @Override
    public double cost() {
        return 0.5 + beverage.cost(); // 糖的价格加上原始饮料的价格
    }
}
  1. 测试装饰者模式
java 复制代码
public class CoffeeShop {
    public static void main(String[] args) {
        // 创建一杯基本的咖啡
        Beverage beverage = new Coffee();
        System.out.println(beverage.getDescription() + " $" + beverage.cost());

        // 给咖啡加牛奶
        beverage = new Milk(beverage);
        System.out.println(beverage.getDescription() + " $" + beverage.cost());

        // 给咖啡加牛奶和糖
        beverage = new Sugar(beverage);
        System.out.println(beverage.getDescription() + " $" + beverage.cost());
    }
}

输出结果:

java 复制代码
Coffee $5.0
Coffee, Milk $6.5
Coffee, Milk, Sugar $7.0
相关推荐
桦说编程34 分钟前
从 ForkJoinPool 的 Compensate 看并发框架的线程补偿思想
java·后端·源码阅读
躺平大鹅2 小时前
Java面向对象入门(类与对象,新手秒懂)
java
静水流深_沧海一粟3 小时前
04 | 别再写几十个参数的构造函数了——建造者模式
设计模式
StarkCoder3 小时前
从UIKit到SwiftUI的迁移感悟:数据驱动的革命
设计模式
初次攀爬者3 小时前
RocketMQ在Spring Boot上的基础使用
java·spring boot·rocketmq
花花无缺3 小时前
搞懂@Autowired 与@Resuorce
java·spring boot·后端
Derek_Smart5 小时前
从一次 OOM 事故说起:打造生产级的 JVM 健康检查组件
java·jvm·spring boot
NE_STOP6 小时前
MyBatis-mybatis入门与增删改查
java
孟陬9 小时前
国外技术周刊 #1:Paul Graham 重新分享最受欢迎的文章《创作者的品味》、本周被划线最多 YouTube《如何在 19 分钟内学会 AI》、为何我不
java·前端·后端