2.装饰器模式

时间:2025-03-10 07:35:14

概述

装饰器模式:在原有结构,动态地为对象添加职责,它是一种灵活的扩展功能方式。

业务场景:创建订单

假设你正在开发一个电商系统,用户在创建订单时可以选择不同的服务(如折扣、配送、礼品包装等)。你需要灵活地计算订单的总价,并能够动态地添加或移除这些服务。

核心思想

在创建订单场景中的核心思想是动态扩展,它通过组合而非继承的方式,创建时能不断扩展,订单对象传进入参不断扩展。


1. 基础组件接口 (Order)

public interface Order {
    double getTotalPrice(); // 获取订单总价
    String getDescription(); // 获取订单描述
}

2. 具体组件 (BasicOrder)

public class BasicOrder implements Order {
    private double basePrice; // 订单基础价格

    public BasicOrder(double basePrice) {
        this.basePrice = basePrice;
    }

    @Override
    public double getTotalPrice() {
        return basePrice; // 返回基础价格
    }

    @Override
    public String getDescription() {
        return "基础订单(价格:" + basePrice + ")";
    }
}

3. 装饰器基类 (OrderDecorator)

public abstract class OrderDecorator implements Order {
    protected Order decoratedOrder;

    public OrderDecorator(Order order) {
        this.decoratedOrder = order;
    }

    @Override
    public double getTotalPrice() {
        return decoratedOrder.getTotalPrice();
    }

    @Override
    public String getDescription() {
        return decoratedOrder.getDescription();
    }
}

4. 具体装饰器 

折扣装饰器 DiscountDecorator
public class DiscountDecorator extends OrderDecorator {
    private double discountRate; // 折扣率

    public DiscountDecorator(Order order, double discountRate) {
        super(order);
        this.discountRate = discountRate;
    }

    @Override
    public double getTotalPrice() {
        return super.getTotalPrice() * (1 - discountRate); // 应用折扣
    }

    @Override
    public String getDescription() {
        return super.getDescription() + " + 折扣(" + (discountRate * 100) + "%)";
    }
}
加急配送装饰器 ExpressShippingDecorator
public class ExpressShippingDecorator extends OrderDecorator {
    private double expressShippingFee; // 加急配送费用

    public ExpressShippingDecorator(Order order, double expressShippingFee) {
        super(order);
        this.expressShippingFee = expressShippingFee;
    }

    @Override
    public double getTotalPrice() {
        return super.getTotalPrice() + expressShippingFee; // 添加加急配送费用
    }

    @Override
    public String getDescription() {
        return super.getDescription() + " + 加急配送(费用:" + expressShippingFee + ")";
    }
}
礼品包装装饰器 GiftWrapDecorator
public class GiftWrapDecorator extends OrderDecorator {
    private double giftWrapFee; // 礼品包装费用

    public GiftWrapDecorator(Order order, double giftWrapFee) {
        super(order);
        this.giftWrapFee = giftWrapFee;
    }

    @Override
    public double getTotalPrice() {
        return super.getTotalPrice() + giftWrapFee; // 添加礼品包装费用
    }

    @Override
    public String getDescription() {
        return super.getDescription() + " + 礼品包装(费用:" + giftWrapFee + ")";
    }
}

5. 客户端代码

public class OrderSystem {
    public static void main(String[] args) {
        // 创建一个基础订单
        Order order = new BasicOrder(100.0);
        System.out.println(order.getDescription());
        System.out.println("总价: " + order.getTotalPrice());

        // 添加折扣
        order = new DiscountDecorator(order, 0.1); // 10% 折扣
        System.out.println(order.getDescription());
        System.out.println("总价: " + order.getTotalPrice());

        // 添加加急配送
        order = new ExpressShippingDecorator(order, 15.0); // 加急配送费用 15
        System.out.println(order.getDescription());
        System.out.println("总价: " + order.getTotalPrice());

        // 添加礼品包装
        order = new GiftWrapDecorator(order, 5.0); // 礼品包装费用 5
        System.out.println(order.getDescription());
        System.out.println("总价: " + order.getTotalPrice());
    }
}

6. 输出

基础订单(价格:100.0)
总价: 100.0
基础订单(价格:100.0) + 折扣(10.0%)
总价: 90.0
基础订单(价格:100.0) + 折扣(10.0%) + 加急配送(费用:15.0)
总价: 105.0
基础订单(价格:100.0) + 折扣(10.0%) + 加急配送(费用:15.0) + 礼品包装(费用:5.0)
总价: 110.0

业务场景总结

  1. 问题:订单可能需要添加多种附加服务(如折扣、运费、包装费等),如果为每种组合创建单独的类,会导致类爆炸。

  2. 解决方案:使用装饰器模式,动态地为订单添加附加服务。

  3. 优点

    • 灵活扩展功能,无需修改订单类的核心逻辑。

    • 支持动态添加或移除服务,符合开闭原则。

    • 避免类爆炸,组合功能更加简洁。


其他业务场景

  1. 购物车:为购物车动态添加优惠券、满减活动等。

  2. 订阅服务:为用户订阅动态添加附加功能(如高级功能、额外存储等)。

  3. 账单系统:为账单动态添加税费、服务费等。

装饰器模式在创建订单场景中的核心思想是动态扩展,它通过组合而非继承的方式,灵活地为订单添加功能或服务。