趣解装饰者模式之《我想吃煎饼果子了》

〇、小故事

话说最近早起没时间做早饭,并且早上上班的地铁口不远处就有一处非常火爆的煎饼摊,所以我就经常去那边吃煎饼,一个"基础版"煎饼是7块钱,向煎饼中加一颗鸡蛋是1元钱加一根火腿肠是3元钱加鸡柳是4元钱......

好像基本上能想到的美食都能往煎饼里塞似的。这就让我想起之前看过的一个短视频,一个小伙子去买煎饼,说要加50颗鸡蛋,然后摊出来的煎饼果子跟一个大披萨似的,非常有趣。

null

那么,做煎饼果子似乎没什么难度,但是,怎么计算总价呢? 我们可以往里放各种"辅料",那计算出来的总价也会千差万别,如果说我们针对不同的煎饼搭配组合都实现cost()价格方法,这样可以吗?如果这种种类少,其实是可以的,但是,如果组合非常多,那必然就造成了"类爆炸了",那么这种设计方式就不合适了。

null

那我们还有什么解决办法吗?我们可以采用装饰者模式,来解决这个问题。下面,我们就将视野转到装饰者模式吧。

一、模式定义

装饰者模式定义:

动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。

模拟顾客购买煎饼果子+1颗鸡蛋+1跟香肠的过程:

null

二、模式类图

为了可以进一步说明装饰者模式在购买煎饼问题上的处理方法,我们以类图的方式展示煎饼与辅料之间的关系。需要说明的一点是,为了表示煎饼的实习类可以是一个"种族",所以列举出了"白面煎饼"、"黑面煎饼"和"杂粮煎饼",这些都属于上面我们所描述的基础类煎饼

null

三、代码实现

创建食物抽象类Food,所有食物都需要继承该抽象类:

csharp 复制代码
public abstract class Food {
    String description = "食物";

    public String getDescription() {
        return description;
    }

    public abstract double cost();
}

创建辅料抽象类AccessoriesDecorator,后续添加到煎饼果子里的辅料都需要继承该类:

csharp 复制代码
public abstract class AccessoriesDecorator extends Food {
    public abstract String getDescription();
}

创建基础版煎饼果子类JianBing ,需要继承Food抽象类:

csharp 复制代码
public class JianBing extends Food {

    public JianBing() {
        description = "基础版煎饼果子";
    }

    public double cost() {
        System.out.println("基础版煎饼果子售价:" + 7 + "元");
        return 7;
    }
}

创建鸡蛋类Egg ,由于它属于辅料,所以继承AccessoriesDecorator抽象类:

csharp 复制代码
public class Egg extends AccessoriesDecorator {

    private Food food;

    public Egg(Food food) {
        this.food = food;
    }

    public String getDescription() {
        return "鸡蛋";
    }

    public double cost() {
        System.out.println("添加鸡蛋售价:" + 1 + "元");
        return 1 + food.cost();
    }
}

创建鸡蛋类Sausage ,由于它属于辅料,所以继承AccessoriesDecorator抽象类:

csharp 复制代码
public class Sausage extends AccessoriesDecorator {
    private Food food;

    public Sausage(Food food) {
        this.food = food;
    }

    public String getDescription() {
        return "香肠";
    }

    public double cost() {
        System.out.println("添加香肠售价:" + 3 + "元");
        return 3 + food.cost();
    }
}

创建测试类DecoratorTest ,计算购买1个基础版煎饼果子加入1颗鸡蛋加入1根香肠的总消费金额:

java 复制代码
public class DecoratorTest {
    public static void main(String[] args) {
        // 创建煎饼果子
        Food jianBing = new JianBing();
        
        // 创建鸡蛋,加入到煎饼果子中
        Food egg = new Egg(jianBing);
        
        // 创建香肠,加入到煎饼果子中
        Food sausage = new Sausage(egg);
        
        System.out.println("总金额为:" + sausage.cost() + "元");
    }
}

输出结果:

复制代码
添加香肠售价:3元
添加鸡蛋售价:1元
基础版煎饼果子售价:7元
总金额为:11.0元

今天的文章内容就这些了:

写作不易,笔者几个小时甚至数天完成的一篇文章,只愿换来您几秒钟的 点赞 & 分享

更多技术干货,欢迎大家关注公众号"爪哇缪斯" ~ (^o^)/ ~ 「干货分享,每天更新」

相关推荐
海南java第二人1 天前
Spring MVC核心流程深度解析:从请求到响应的完美掌控
java·springmvc
未来之窗软件服务1 天前
幽冥大陆(一百10)PHP打造Java的Jar安全——东方仙盟筑基期
java·php·phar·仙盟创梦ide·东方仙盟
程序猿_极客1 天前
【2025 年最新版】Java JDK 安装与环境配置教程(附图文超详细,Windows+macOS 通用)
java·开发语言·windows·macos·jdk
猫头虎1 天前
macOS 双开/多开微信WeChat完整教程(支持 4.X 及以上版本)
java·vscode·macos·微信·编辑器·mac·脚本
二哈喇子!1 天前
Java开发工具——IDEA(修改全局配置,提升工作效率)
java·编辑器·intellij-idea
强子感冒了1 天前
Java网络编程学习笔记,从网络编程三要素到TCP/UDP协议
java·网络·学习
咚为1 天前
Rust Print 终极指南:从底层原理到全场景实战
开发语言·后端·rust
二哈喇子!1 天前
SpringBoot项目右上角选择ProjectNameApplication的配置
java·spring boot
sin22011 天前
MyBatis的执行流程
java·开发语言·mybatis
二哈喇子!1 天前
基于Spring Boot框架的车库停车管理系统的设计与实现
java·spring boot·后端·计算机毕业设计