您的位置:首页 > 健康 > 美食 > 设计模式大白话之装饰者模式

设计模式大白话之装饰者模式

2024/10/6 10:19:14 来源:https://blog.csdn.net/qq_36634055/article/details/140380948  浏览:    关键词:设计模式大白话之装饰者模式

想象一下,你走进一家咖啡馆,点了一杯美式咖啡。但是,你可能还想根据自己的口味添加一些东西,比如奶泡、巧克力粉、焦糖酱或是肉桂粉。每次你添加一种配料,你的咖啡就会变得更丰富,同时价格也会相应增加。

在软件设计中,装饰者模式就是让你能够在不改变原始对象的情况下,动态地给它添加新功能或增强现有功能。这就好比你在咖啡上加配料,不是通过制作一种全新的咖啡,而是通过在现有的基础上增加东西。

比如,假设你正在开发一个文本编辑器,你希望用户可以选择开启一些额外的功能,比如语法高亮、行号显示、自动完成等。你不想为每一种可能的组合都创建一个单独的文本编辑器类,因为那样会有无数种组合,维护起来会非常困难。

这时,装饰者模式就派上用场了。你可以创建一个基本的文本编辑器类,然后为每一个额外功能创建一个装饰者类。当用户选择开启某项功能时,你就在基本的文本编辑器上“装饰”上相应的装饰者。这样,用户就可以根据自己的需求,动态地定制他们自己的文本编辑器,而不需要重新编写整个程序。

简而言之,装饰者模式就像是给你的软件产品添加“配料”,让它变得更有个性,更符合用户的需求,而无需更改其核心结构。

 以一家奶茶店为例,使用装饰者模式来动态地给奶茶添加各种配料。我们将创建一个基础的奶茶类,然后使用装饰者类来添加不同的配料,如珍珠、布丁、红豆、仙草、芝士等。

首先,我们定义奶茶的基本接口:
public abstract class Drink {// 基本属性protected String description = "Unknown Drink";// 获取描述public String getDescription() {return description;}// 设置描述public void setDescription(String description) {this.description = description;}// 抽象方法,计算价格public abstract double cost();
}
接下来,我们创建基础的奶茶类:
public class MilkTea extends Drink {public MilkTea() {description = "Milk Tea";}public double cost() {return 3.5; // 假设一杯基础奶茶的价格是3.5元}
}
然后,我们定义装饰者基类:
public abstract class AddOns extends Drink {protected Drink drink;public AddOns(Drink drink) {this.drink = drink;}public String getDescription() {return drink.getDescription();}public double cost() {return drink.cost();}
}
接着,我们创建具体的配料装饰者:
public class Boba extends AddOns {public Boba(Drink drink) {super(drink);}@Overridepublic String getDescription() {// 返回被装饰对象的描述return super.getDescription()+",  with Boba";}@Overridepublic double cost() {return drink.cost() + 1.0; // 添加珍珠的成本}
}
public class Pudding extends AddOns {public Pudding(Drink drink) {super(drink);}@Overridepublic String getDescription() {// 返回被装饰对象的描述return super.getDescription()+", with Pudding";}@Overridepublic double cost() {return drink.cost() + 0.8; // 添加布丁的成本}
}
public class RedBeans extends AddOns {public RedBeans(Drink drink) {super(drink);}@Overridepublic String getDescription() {// 返回被装饰对象的描述return super.getDescription()+",  with Red Beans";}@Overridepublic double cost() {return drink.cost() + 0.7; // 添加红豆的成本}
}
最后,在主函数中使用这些装饰者:
public class Main {public static void main(String[] args) {Drink milkTea = new MilkTea();System.out.println(milkTea.getDescription() + ": " + milkTea.cost());Drink milkTeaWithBoba = new Boba(milkTea);System.out.println(milkTeaWithBoba.getDescription() + ": " +  milkTeaWithBoba.cost());Drink milkTeaWithBobaAndPudding = new Pudding(milkTeaWithBoba);System.out.println(milkTeaWithBobaAndPudding.getDescription() + ": " + milkTeaWithBobaAndPudding.cost());Drink milkTeaWithBobaAndRedBeans = new RedBeans(milkTeaWithBoba);System.out.println(milkTeaWithBobaAndRedBeans.getDescription() + ": " + milkTeaWithBobaAndRedBeans.cost());}
}
执行结果
Milk Tea: 3.5
Milk Tea,  with Boba: 4.5
Milk Tea,  with Boba, with Pudding: 5.3
Milk Tea,  with Boba,  with Red Beans: 5.2

在上述代码中,Drink 是奶茶及其配料的抽象基类,MilkTea 是基本的奶茶实现,AddOns 是所有配料的基类,而Boba, Pudding, RedBeans 分别是具体的配料装饰者。通过组合不同的装饰者,我们可以轻松地构建出各种不同配料组合的奶茶,而且这些组合是在运行时动态决定的。

理解这种思想然后之后就是如何熟练运用,在有需要的时候。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com