java设计模式-结构型模式:装饰者模式

定义:

在不改变原有对象的基础之上,将功能附加到对象上,提供了比继承更有弹性的替代方案。

适用于:

  1. 拓展一个类的功能;
  2. 动态给对象添加功能,并且动态撤销。

优点:

  1. 继承的有力补充,不改变原有对象的情况下给对象拓展功能;
  2. 通过使用不同的装饰类、不同的组合方式,实现不同的效果。
  3. 符合开闭原则。

缺点:

  1. 增加程序复杂性;

举个水果沙拉的例子。

比如在点水果沙拉外卖时,可以往水果沙拉里加各种水果,价格也会相应的调整,要让程序支持不同水果自由组合,并计算相应的价格,则可以使用装饰者模式来完成。

定义一个抽象的水果沙拉类AbstractFruitSalad:

public abstract class AbstractFruitSalad {
 public abstract String remark();
 public abstract int price();
}

包含备注和价格抽象方法。

接着创建一个抽象的装饰器AbstractDecorator(关键点,继承抽象水果沙拉类):

public class AbstractDecorator extends AbstractFruitSalad{

 private AbstractFruitSalad fruitSalad;

 public AbstractDecorator(AbstractFruitSalad fruitSalad){
 this.fruitSalad = fruitSalad;
 }

 @Override
 public String remark() {
 return fruitSalad.remark();
 }

 @Override
 public int price() {
 return fruitSalad.price();
 }
}

创建具体的水果沙拉类FruitSalad:

public class FruitSalad extends AbstractFruitSalad{
 @Override
 public String remark() {
 return "水果🥗(标准)\n";
 }

 @Override
 public int price() {
 return 9;
 }
}

该沙拉是标准的水果沙拉,价格是9元。

如果我们的水果沙拉还允许客户添加猕猴桃和西瓜,那么我们可以添加两个新的装饰器。添加猕猴桃装饰器KiwiDecorator:

public class KiwiDecorator extends AbstractDecorator {

 public KiwiDecorator(AbstractFruitSalad fruitSalad) {
 super(fruitSalad);
 }

 @Override
 public String remark() {
 return super.remark() + "加份🥝切\n";
 }

 @Override
 public int price() {
 return super.price() + 2;
 }
}

可以看到,加一份猕猴桃需要在原有基础上加2元。

接着继续创建西瓜装饰器WaterMelonDecorator:

public class WaterMelonDecorator extends AbstractDecorator {

 public WaterMelonDecorator(AbstractFruitSalad fruitSalad) {
 super(fruitSalad);
 }

 @Override
 public String remark() {
 return super.remark() + "加份🍉切\n";
 }

 @Override
 public int price() {
 return super.price() + 3;
 }
}

最后创建客户端Application测试一下:

public class Application {

 public static void main(String[] args) {
 // 点了份水果沙拉,并加了两份🥝和一份🍉,看看最终价格是多少?
 AbstractFruitSalad fruitSalad = new FruitSalad();
 fruitSalad = new KiwiDecorator(fruitSalad);
 fruitSalad = new KiwiDecorator(fruitSalad);
 fruitSalad = new WaterMelonDecorator(fruitSalad);

 System.out.println(fruitSalad.remark() + "价格是:" + fruitSalad.price());
 }
}

上面的写法也可以改为:

public class Application {

 public static void main(String[] args) {
 // 点了份水果沙拉,并加了两份🥝和一份🍉,看看最终价格是多少?
 AbstractFruitSalad fruitSalad = new FruitSalad();
 fruitSalad = new WaterMelonDecorator(new KiwiDecorator(new KiwiDecorator(fruitSalad)));

 System.out.println(fruitSalad.remark() + "价格是:" + fruitSalad.price());
 }
}

程序输出如下:

水果🥗(标准)
加份🥝切
加份🥝切
加份🍉切
价格是:16

通过不同的装饰器自由组合,我们可以灵活的组装出各式各样的水果沙拉,这正是装饰者模式的优点,但明显可以看出代码变复杂了。

这个例子的UML图如下所示:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 装饰者模式的定义 装饰者模式动态地将责任附加到对象上,若要扩展功能,装饰者提供了比继承更有弹性的替代方案。 可以这...
    lhsjohn阅读 3,110评论 0 0
  • 大部分公司都有销售团队,假设老板给你布置了一个任务,让你按照下面的要求开发一套程序来计算销售团队每个月的工资。 每...
    西木柚子阅读 4,645评论 2 11
  • 装饰者模式的应用场景 装饰者模式(DecoratorPattern)是指在不改变原有对象的基础之上,将功能附加到对...
    提玛欧斯阅读 1,898评论 0 0
  • 定义:在不必改变原类文件和原使用的继承的情况下,动态地扩展一个对象的功能。它是通过创建一个包装对象,也就是用装饰来...
    holy_z阅读 2,767评论 0 0
  • 推荐指数: 6.0 书籍主旨关键词:特权、焦点、注意力、语言联想、情景联想 观点: 1.统计学现在叫数据分析,社会...
    Jenaral阅读 10,997评论 0 5