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

〇、小故事

话说最近早起没时间做早饭,并且早上上班的地铁口不远处就有一处非常火爆的煎饼摊,所以我就经常去那边吃煎饼,一个"基础版"煎饼是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^)/ ~ 「干货分享,每天更新」

相关推荐
kangkang-20 分钟前
PC端基于SpringBoot架构控制无人机(三):系统架构设计
java·架构·无人机
界面开发小八哥2 小时前
「Java EE开发指南」如何用MyEclipse创建一个WEB项目?(三)
java·ide·java-ee·myeclipse
ai小鬼头2 小时前
Ollama+OpenWeb最新版0.42+0.3.35一键安装教程,轻松搞定AI模型部署
后端·架构·github
idolyXyz2 小时前
[java: Cleaner]-一文述之
java
一碗谦谦粉3 小时前
Maven 依赖调解的两大原则
java·maven
萧曵 丶3 小时前
Rust 所有权系统:深入浅出指南
开发语言·后端·rust
netyeaxi3 小时前
Java:使用spring-boot + mybatis如何打印SQL日志?
java·spring·mybatis
收破烂的小熊猫~3 小时前
《Java修仙传:从凡胎到码帝》第四章:设计模式破万法
java·开发语言·设计模式
猴哥源码3 小时前
基于Java+SpringBoot的动物领养平台
java·spring boot
老任与码4 小时前
Spring AI Alibaba(1)——基本使用
java·人工智能·后端·springaialibaba