正文

一、定义

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

要点:

  • 装饰者和被装饰者有相同的超类型。
  • 可以用一个或多个装饰者包装一个对象。
  • 既然装饰者和被装饰者有相同的超类型,所以在任何需要原始对象(被装饰者)的场合,都可以用装饰过的对象代替它。
  • 装饰者可以在被装饰者的行为之前与/或之后,加上自己的行为,甚至将被装饰者的行为整个取代掉,以到达特定的目的。
  • 对象可以在任何时候被装饰,所以可以在运行时动态地、不限量地用装饰者装饰对象。
  • 装饰者会导致设计中出现许多小对象,如果过度使用,会让程序变得很复杂。

二、实现步骤

1、创建组件接口

装饰者和被装饰者都必须实现组件接口。

也可以用组件抽象类,然后让装饰者和被装饰者继承组件抽象类,只要装饰者和被装饰者具有相同的超类型即可。

/**
* 组件接口(装饰者和被装饰者都必须实现该接口)
*/
public interface Component { public void doSomething();
}

2、创建具体的组件,并实现组件接口

/**
* 具体组件(被装饰者)
*/
public class ConcreteComponent implements Component { @Override
public void doSomething() {
System.out.println("ConcreteComponent do something...");
}
}

3、创建装饰者抽象类,并实现组件接口

如果只有一个装饰者,也可以不创建装饰者抽象类,而是由具体的装饰者直接实现组件接口

/**
* 组件装饰者抽象类
*/
public abstract class ComponentDecorator implements Component { protected Component component; public ComponentDecorator(Component component) {
// 通过构造传入组件(被装饰者)
this.component = component;
} @Override
public void doSomething() {
// 委托给组件(被装饰者)
component.doSomething();
}
}

4、创建具体的装饰者,并继承装饰者抽象类

(1)装饰者 A

/**
* 装饰者A
*/
public class ComponentDecoratorA extends ComponentDecorator { public ComponentDecoratorA(Component component) {
super(component);
} @Override
public void doSomething() {
// 装饰者添加自己的业务代码 component.doSomething(); // 装饰者添加自己的业务代码
System.out.println("ComponentDecoratorA do something...");
}
}

(2)装饰者 B

/**
* 装饰者B
*/
public class ComponentDecoratorB extends ComponentDecorator { public ComponentDecoratorB(Component component) {
super(component);
} @Override
public void doSomething() {
// 装饰者添加自己的业务代码 component.doSomething(); // 装饰者添加自己的业务代码
System.out.println("ComponentDecoratorB do something...");
}
}

5、使用装饰者装饰组件

public class Test {

    public static void main(String[] args) {
// 具体组件(被装饰者)
Component component = new ConcreteComponent();
// 用装饰者A装饰组件
ComponentDecorator componentDecoratorA = new ComponentDecoratorA(component);
// 用装饰者B装饰组件
ComponentDecorator componentDecoratorB = new ComponentDecoratorB(component); component.doSomething();
componentDecoratorA.doSomething();
componentDecoratorB.doSomething();
}
}

三、举个栗子

1、背景

星巴兹是以扩张速度最快而闻名的咖啡连锁店。因为扩张速度实在太快了,他们准备更新订单系统,以合乎他们的饮料供应要求——

顾客在购买咖啡时,可以要求在其中加入各种调料,例如:蒸奶、豆浆、摩卡(巧克力风味)或覆盖奶泡。星巴兹会根据所加入的调料收取不同的费用。所以订单系统必须考虑到这些调料部分。

2、实现

把调料理解为饮料装饰者,然后以饮料为主体,用调料来“装饰”饮料。

(1)创建饮料抽象类

/**
* 饮料抽象类(组件)
*/
public abstract class Beverage { public String description = "Unknown Beverage"; /**
* 描述
*/
public String getDescription() {
return description;
} /**
* 价格
*/
public abstract double cost();
}

(2)创建具体的饮料,并继承饮料抽象类

/**
* 浓缩咖啡
*/
public class Espresso extends Beverage { public Espresso() {
description = "Espresso";
} @Override
public double cost() {
return 1.99;
}
}
/**
* 综合咖啡
*/
public class HouseBlend extends Beverage { public HouseBlend() {
description = "House Blend Coffee";
} @Override
public double cost() {
return 0.89;
}
}

(3)创建调料抽象类,并继承饮料抽象类

/**
* 调料抽象类(装饰者抽象类)
*/
public abstract class CondimentDecorator extends Beverage { @Override
public abstract String getDescription();
}

(4)创建具体的调料,并继承调料抽象类

/**
* 摩卡(装饰者)
*/
public class Mocha extends CondimentDecorator { Beverage beverage; public Mocha(Beverage beverage) {
this.beverage = beverage;
} @Override
public String getDescription() {
return beverage.getDescription() + ", Mocha";
} @Override
public double cost() {
// 加上摩卡的价格
return beverage.cost() + 0.20;
}
}
/**
* 豆浆(装饰者)
*/
public class Soy extends CondimentDecorator { Beverage beverage; public Soy(Beverage beverage) {
this.beverage = beverage;
} @Override
public String getDescription() {
return beverage.getDescription() + ", Soy";
} @Override
public double cost() {
// 加上豆浆的价格
return beverage.cost() + 0.15;
}
}
/**
* 奶泡(装饰者)
*/
public class Whip extends CondimentDecorator { Beverage beverage; public Whip(Beverage beverage) {
this.beverage = beverage;
} @Override
public String getDescription() {
return beverage.getDescription() + ", Whip";
} @Override
public double cost() {
// 加上奶泡的价格
return beverage.cost() + 0.10;
}
}

(5)测试

public class Test {

    public static void main(String[] args) {
// 浓缩咖啡
Beverage beverage = new Espresso();
System.out.println(beverage.getDescription() + " $" + beverage.cost()); // 综合咖啡
Beverage beverage2 = new HouseBlend();
System.out.println(beverage2.getDescription() + " $" + beverage2.cost());
// 添加摩卡
beverage2 = new Mocha(beverage2);
// 添加豆浆
beverage2 = new Soy(beverage2);
// 添加奶泡
beverage = new Whip(beverage2);
System.out.println(beverage2.getDescription() + " $" + beverage2.cost());
}
}

《Head First 设计模式》:装饰者模式的更多相关文章

  1. Java设计模式——装饰者模式

    JAVA 设计模式 装饰者模式 用途 装饰者模式 (Decorator) 动态地给一个对象添加一些额外的职责.就增加功能来说,Decorator 模式相比生成子类更为灵活. 装饰者模式是一种结构式模式 ...

  2. JAVA设计模式--装饰器模式

    装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构.这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰 ...

  3. 从源码角度理解Java设计模式——装饰者模式

    一.饰器者模式介绍 装饰者模式定义:在不改变原有对象的基础上附加功能,相比生成子类更灵活. 适用场景:动态的给一个对象添加或者撤销功能. 优点:可以不改变原有对象的情况下动态扩展功能,可以使扩展的多个 ...

  4. 【设计模式】Java设计模式 - 装饰者模式

    Java设计模式 - 装饰者模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起记录分享自 ...

  5. [Head First设计模式]山西面馆中的设计模式——装饰者模式

    引言 在山西面馆吃鸡蛋面的时候突然想起装饰者这个模式,觉得面馆这个场景跟书中的星巴兹咖啡的场景很像,边吃边思考装饰者模式.这里也就依葫芦画瓢,换汤不换药的用装饰者模式来模拟一碗鸡蛋面是怎么出来的吧.吃 ...

  6. JAVA 设计模式 装饰者模式

    用途 装饰者模式 (Decorator) 动态地给一个对象添加一些额外的职责.就增加功能来说,Decorator 模式相比生成子类更为灵活. 装饰者模式是一种结构式模式. 结构

  7. 浅谈设计模式--装饰者模式(Decorator Pattern)

    挖了设计模式这个坑,得继续填上.继续设计模式之路.这次讨论的模式,是 装饰者模式(Decorator Pattern) 装饰者模式,有时也叫包装者(Wrapper),主要用于静态或动态地为一个特定的对 ...

  8. javascript设计模式——装饰者模式

    前面的话 在程序开发中,许多时候都并不希望某个类天生就非常庞大,一次性包含许多职责.那么可以使用装饰者模式.装饰者模式可以动态地给某个对象添加一些额外的职责,而不会影响从这个类中派生的其他对象.本文将 ...

  9. 设计模式-装饰者模式(Decorator Pattern)

    本文由@呆代待殆原创,转载请注明出处. 此设计模式遵循的设计原则之一:类应该支持扩展,而拒绝修改(Open-Closed Principle) 装饰者模式简述 装饰者模式通过组合的方式扩展对象的特性, ...

  10. C#设计模式--装饰器模式

    0.C#设计模式-简单工厂模式 1.C#设计模式--工厂方法模式 2.C#设计模式--抽象工厂模式 3.C#设计模式--单例模式 4.C#设计模式--建造者模式 5.C#设计模式--原型模式 6.C# ...

随机推荐

  1. 2019-02-04 Linux的一些常用命令学习

    今天在电脑里用装了个ubuntu的虚拟机,学习了一下基本操作 ls ls用于查看文件夹的文件 mkdir mkdir可以创建一个文件夹(directory) cd 目录切换 pwd print wor ...

  2. TensorFlow从0到1之常量、变量和占位符详解(6)

    最基本的 TensorFlow 提供了一个库来定义和执行对张量的各种数学运算.张量,可理解为一个 n 维矩阵,所有类型的数据,包括标量.矢量和矩阵等都是特殊类型的张量.   TensorFlow 支持 ...

  3. Spark原始码系列(六)Shuffle的过程解析

      问题导读: 1.shuffle过程的划分? 2.shuffle的中间结果如何存储? 3.shuffle的数据如何拉取过来? Shuffle过程的划分 Spark的操作模型是基于RDD的,当调用RD ...

  4. Java收徒,高级架构师关门弟子

    最近感悟天命,偶有所得,故而打算收徒若干,以继吾之传承. 有缘者,可破瓶颈,走向架构师之峰,指日可待. 拜师要求: 1.工作经验:1年或以上. 2.入门费用:10000元(RMB). 联系方式(联系时 ...

  5. 税务ukey如何批量开票

    最近税局开始大力推税务ukey版本,不过目前接口还未开放,就连航信,百旺否还没有对应接口,所以自己研究了下,在之前税控基础上,谁知道搞定了,通过安装插件可以批量开票,包括纸质,电子发票ofd格式. 联 ...

  6. IP地址和端口

    IP地址是网络中计算机的唯一标识.没有IP地址,计算机无法接入互联网. IPv4地址32bit,用点分十进制表示,如202.38.64.3 IPv6地址128bit,用冒号分割十六进制表示,如2001 ...

  7. 三角函数与缓入缓出动画及C#实现(图文讲解)

    日常经常能看到缓入缓出的动画效果,如: 1,带缓入缓出效果的滚动条: 2,带缓入缓出效果的呼吸灯: 像上面这种效果,就是用到了三角函数相关的知识,下面将从头开始一步步去讲解如何实现这种效果. 一.基础 ...

  8. Jmeter各种组件

    断言 用于检查测试中得到的响应数据等是否符合预期,用以保证性能测试过程中的数据交互与预期一致 参数化关联 参数化:指对每次发起的请求,参数名称相同,参数值进行替换,如登录三次系统,每次用不同的用户名和 ...

  9. 10TB级日志的秒级搜索

  10. 黎活明8天快速掌握android视频教程--24_网络通信之网页源码查看器

    1 该项目的主要功能就是从将后台的html网页在Android的界面上显示出来 后台就是建立一个java web工程在工程尚建立一个html或者jsp文件就可以了,这里主要看Android客户端的程序 ...