`
mojinglf
  • 浏览: 13054 次
  • 性别: Icon_minigender_1
  • 来自: 大连
最近访客 更多访客>>
社区版块
存档分类
最新评论

初学Java设计模式随记 -- 装饰(Decorator)模式

阅读更多

装饰(Decorator)模式又叫包装器(Wrapper)模式。

 

1. 用意
动态地给一个对象添加一些额外的职责。

 

2. 参与者
• 抽象构件(Component):定义一个对象接口,可以给这些对象动态地添加职责。
• 具体构件(Concrete Component):定义一个对象,可以给这个对象添加一些职责。
• 抽象装饰者(Decorator):持有一个构件(Component)对象的实例,并定义一个与抽象构件接口一致的接口。

• 具体装饰者(Concrete Decorator):向组件添加职责。

 

3.结构


 


 在阎宏的设计模式中,提到“装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方法。”

 

先来理解装饰模式的用意-----------动态地给一个对象添加一些额外的职责。 

所谓动态是说可以在系统运行时(RunTime)动态给对象增加其它职责而不需要修改代码或重新编译;

所谓静态是说必须通过调整代码(DesignTime)才能给对象增加职责,而且系统还需要重新编译;

从具体技术层面来说,对象的组合和继承正好对应于前面的动态和静态

我们应该多使用对象组合来保持系统的运行时扩展性,尽量少使继承,因为继承让程序变得僵硬!

这就是优先使用对象组合,而非类继承(Favor composition over inheritance.)。

 

 

再来理解一下“装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方法。”这句话。 

Decorator是装饰者模式里非常特殊的一个类,它既继承于Component(IS A关系),又维护一个指向Component实例的引用(HAS A关系),换个角度来说,Decorator跟Component之间,既有动态组合关系又有静态继承关系。

这里为什么要这么来设计?

组合的好处是可以在运行时给对象增加职责,Decorator 有一个(HAS A) Component的目的是让具体装饰者(Concrete Decorator)可以在运行时动态给具体构件(Concrete Component)增加职责,这一点相对来说还比较好理解; 

那么Decorator继承于Component的目的是什么?

在这里,继承的目的只有一个,那就是可以统一装饰者和被装饰者的接口,换个角度来说,不管是具体构件(Concrete Component)还是具体装饰者(Concrete Decorator),它们都是 Component,用户代码可以把它们统一看作Component来处理,这样带来的更深一层的好处就是,装饰者对象被装饰者对象的功能职责扩展对用户代码来说是完全透明的,因为用户代码引用的都是Component,所以就不会因为被装饰者对象在被装饰后,引用它的用户代码发生错误,实际上不会有任何影响,因为装饰前后,用户代码引用的都是Component类型的对象。

 

装饰者模式通过继承实现统一了装饰者和被装饰者的接口,通过组合获得了在运行时动态扩展被装饰者对象的能力。

 

 

还是通过例子来说明:

首先,不使用装饰模式,想想怎么实现下面的需求。

1.还是以买汽车为例---顾客要去汽车代理商买汽车,代理商提供了一些服务,刚开始是一些售后维修服务; 

2.后来,还提供了零部件供应的服务;然后,又提供了一些咨询服务(代理商解答顾客买车前的一些疑问);最后,又为客户提供了车险的服务。  

3.如果代理商需要根据情况,提供不同的服务组合。

 

(这里就不写代码来实现了)

 

大多数情况下,都会使用继承,这样,主要会有下面的问题:

1. 系统的扩展性很差;

2. 当需要提供不同的服务组合时,需要创建大量的类,造成类爆炸;

 

那么,看看装饰模式是怎么实现上面的需求的。Java代码如下:

 抽象构件(Component): 汽车销售商 

/**
 * 抽象构件(Component)
 * 
 * 汽车销售商
 */
public interface CarSeller {
	
	/*
	 * 销售汽车
	 */
	String sellCars();

}

 

具体构件(Concrete Component):奥迪汽车经销商 

/**
 * 具体构件(Concrete Component)
 * 
 * 奥迪汽车经销商
 */
public class AudiCarAgency implements CarSeller {

	/*
	 * 销售奥迪汽车
	 */
	public String sellCars() {

		System.out.println("销售奥迪汽车");
		
		return "奥迪汽车";
	}

}

 

抽象装饰者(Decorator):提供服务的奥迪销售商 

/**
 * 抽象装饰者(Decorator)
 * 
 * 提供服务的奥迪销售商
 */
public class AudiCarAgencyWithServices implements CarSeller {

	private CarSeller carSeller = null;

	/*
	 * 构造函数
	 */
	public AudiCarAgencyWithServices(CarSeller carSeller) {
		this.carSeller = carSeller;
	}

	/*
	 * 装饰
	 * @see myDecoratorPattern.sellCars.CarSeller#sellCars()
	 */
	public String sellCars() {
		String carName = null;
		carName = carSeller.sellCars();
		return carName;
	}

}

 

具体装饰者(Concrete Decorator):提供维修服务的奥迪销售商 

/**
 * 具体装饰者(Concrete Decorator)
 * 
 * 提供维修服务的奥迪销售商
 *
 */
public class AudiCarAgencyWithMaintenance extends AudiCarAgencyWithServices {
	
	/*
	 * 构造函数
	 */
	public AudiCarAgencyWithMaintenance(CarSeller carSeller) {
		super(carSeller);
	}
	
	/*
	 * 添加了其他的服务
	 */
	public String sellCars(){		
		String carName = super.sellCars();
		System.out.println("提供维修服务");
		return carName;
	}

}

 

具体装饰者(Concrete Decorator):提供零配件供应的奥迪销售商 

/**
 * 具体装饰者(Concrete Decorator)
 * 
 * 提供零配件供应的奥迪销售商
 *
 */
public class AudiCarAgencyWithSparepart extends AudiCarAgencyWithServices {

	/*
	 * 构造函数
	 */
	public AudiCarAgencyWithSparepart(CarSeller carSeller) {
		super(carSeller);
	}
	
	/*
	 * 添加了其他的服务
	 */
	public String sellCars(){
		String carName = super.sellCars();
		System.out.println("提供零配件供应");
		return carName;
	}

}

 

具体装饰者(Concrete Decorator):提供售前咨询服务的奥迪销售商  

/**
 * 具体装饰者(Concrete Decorator)
 * 
 * 提供售前咨询服务的奥迪销售商
 *
 */
public  class AudiCarAgencyWithConsultation extends AudiCarAgencyWithServices {
	
	/*
	 * 构造函数
	 */
	public AudiCarAgencyWithConsultation(CarSeller carSeller){
		super(carSeller);
	}
	
	/*
	 * 添加了其他的服务
	 */
	public String sellCars() {
		System.out.println("提供了售前咨询服务");
		return super.sellCars();
	}

}

 

具体装饰者(Concrete Decorator):提供车险服务的奥迪销售商

 

/**
 * 具体装饰者(Concrete Decorator)
 * 
 * 提供车险服务的奥迪销售商
 *
 */
public class AudiCarAgencyWithInsurance extends AudiCarAgencyWithServices {

	/*
	 * 构造函数
	 */
	public AudiCarAgencyWithInsurance(CarSeller carSeller) {
		super(carSeller);
	}
	
	/*
	 * 添加了其他的服务
	 */
	public String sellCars(){
		
		String carName = super.sellCars();
		System.out.println("提供车险的购买");
		return carName;
	}

}

 

结构如下:

 

 

 

客户端的调用: 

public class Customer {
	
	public static void main(String[] args){
		
		String car = null;
		
		System.out.println("--------------只卖车,没有服务------------------");	
//		汽车经销商
		CarSeller carSeller = new AudiCarAgency();
		car = carSeller.sellCars();
		System.out.println("买到了" + car);
		
		System.out.println("--------------第一种组合-维修服务------------------");		
//		提供了维修服务
		CarSeller carSellerWithMaintenance = new AudiCarAgencyWithMaintenance(carSeller);
		car = carSellerWithMaintenance.sellCars();
		System.out.println("买到了" + car);
		
		System.out.println("--------------第二种组合-添加了零部件供应服务---------");		
//		提供了零部件供应		
		CarSeller carSellerWithSparepart = new AudiCarAgencyWithSparepart(carSellerWithMaintenance);
		car = carSellerWithSparepart.sellCars();
		System.out.println("买到了" + car);

		System.out.println("--------------第三种组合-添加了咨询服务--------------");		
//		提供了咨询服务		
		CarSeller carSellerWithConsultation = new AudiCarAgencyWithConsultation(carSellerWithSparepart);
		car = carSellerWithConsultation.sellCars();	
		System.out.println("买到了" + car);

		System.out.println("--------------第四种组合-添加了车险服务--------------");		
//		提供了车险	
		CarSeller carSellerWithInsurance = new AudiCarAgencyWithInsurance(carSellerWithConsultation);
		car = carSellerWithInsurance.sellCars();
		System.out.println("买到了" + car);
		
		
		System.out.println("#########################################");	
//		还可以有其他的组合方式
		System.out.println("还可以尝试其他形式自由的组合。。。。。。");
		
	}

}

 

 

运行结果:

--------------只卖车,没有服务------------------
销售奥迪汽车
买到了奥迪汽车
--------------第一种组合-维修服务------------------
销售奥迪汽车
提供维修服务
买到了奥迪汽车
--------------第二种组合-添加了零部件供应服务---------
销售奥迪汽车
提供维修服务
提供零配件供应
买到了奥迪汽车
--------------第三种组合-添加了咨询服务--------------
提供了售前咨询服务
销售奥迪汽车
提供维修服务
提供零配件供应
买到了奥迪汽车
--------------第四种组合-添加了车险服务--------------
提供了售前咨询服务
销售奥迪汽车
提供维修服务
提供零配件供应
提供车险的购买
买到了奥迪汽车
#########################################
还可以尝试其他形式自由的组合。。。。。。

 

可以看到使用装饰模式,有如下优点: 

1. 装饰模式与继承关系的目的都是要扩展对象的功能,但是装饰模式可以提供比继承更多的灵活性。

2. 通过使用不同的具体装饰类以及这些装饰类的排列组合,设计师可以使用很少的类,创造出很多不同的行为组合。

 

以上内容部分参照自阎宏的《Java与模式》以及来杯咖啡-装饰者模式(Decorator)实例应用详解

 

如果仔细观察装饰模式的调用(例如:第二种组合-添加了零部件供应服务)时,感觉装饰模式代理模式的作用好像一样(都给一个对象添加了额外的功能);甚至,如果比较这两个模式的结构时,好像也没什么差别。那么,它们到底有什么区别呢?

 

首先,它们的用意是不同的:

代理模式是 控制 对一个对象的访问。

装饰模式是 增加 一个类的职责(功能)。

 

单从它们的结构来看,

使用代理模式时,在访问对象之前或之后也可以添加一些额外的操作,这一点,装饰模式和代理模式差不多,甚至可以说,这两个模式是一样的,都为一个类(一个对象)添加了其他功能。

但如果仔细体会“控制”和“增加”这两个词汇,可能会发现,

代理模式中,所添加的额外的操作,往往为了确定被访问对象者是否能被访问。

而在装饰模式中,增加的这些功能,不会影响原有对象的操作,被装饰的对象一定会被访问的。

 

  • 大小: 6.6 KB
  • 大小: 13.1 KB
分享到:
评论

相关推荐

    NativeJS随记 - 浅析JavaScript Events

    标题中的“NativeJS随记 - 浅析JavaScript Events”表明这篇博客主要讨论的是JavaScript中的事件处理机制。JavaScript事件是Web开发中的重要组成部分,它允许我们响应用户的交互或浏览器的内部变化。在这里,我们将...

    Java.util随记.doc

    Java.util包是Java标准库中的核心包之一,它包含了大量用于处理各种数据结构和集合的类和接口。在这个包中,我们经常会用到`Iterator`和`List`接口,这两个接口在Java编程中扮演着非常重要的角色。 首先,`Iterator...

    初学VB.NET使用心得随记

    在初学者的学习过程中,理解如何有效地使用`OleDbDataAdapter`来更新数据库至关重要。在提供的代码示例中,我们看到了一个简单的VB.NET应用程序,它使用`OleDbDataAdapter`从Access数据库(`.mdb`)中填充`DataSet`,...

    java随记等

    根据提供的文件信息,我们可以分析出该段代码定义了一个名为 `UserRoleType` 的 Java 枚举类,用于描述不同用户角色的权限类型。接下来,我们将详细解释这个枚举类中的各个元素及其含义,并尝试理解其中的一些注释...

    读书笔记:Java并发编程之美阅读随记.zip

    读书笔记:Java并发编程之美阅读随记

    .class文件反编译成.java文件工具:jd-gui(无需安装,下载直接使用)

    《深入理解Java类文件反编译:jd-gui工具详解》 在Java开发过程中,有时我们需要查看或理解已编译的.class文件中的源代码,这时就需要借助反编译工具。jd-gui是一款非常实用的Java反编译软件,它允许开发者直接查看...

    JSP PDF打印 随记 复杂模板设计

    本篇随记主要探讨的是如何在JSP环境中进行PDF打印,并涉及复杂的模板设计。PDF(Portable Document Format)是一种通用的文件格式,常用于生成保持原始文档格式不变的静态文档,适用于打印和共享。 首先,我们需要...

    随记小时光设计书1

    在"随记小时光设计书1"中,我们主要讨论的是用户信息和手账信息的设计,特别是在数据库方面的应用。这个设计涉及到用户信息的多个关键组成部分,包括昵称、用户名、密码、邮箱、手机号以及权限和加密key的管理。下面...

    JAVA核心知识点整理.zip

    11. **设计模式**:常见的23种设计模式(如工厂模式、单例模式、观察者模式等)在Java中都有应用,是提升代码质量和可维护性的关键。 12. **Java EE**:对于Web开发者,Java Enterprise Edition(Java EE)提供了...

    一些有关哈希函数的随记

    这篇随记将探讨哈希函数的基本概念、性质以及在实际应用中的重要性。 哈希函数,也称为散列函数,是一种特殊的算法,它将任意长度的输入(也称为预映射或消息)转化为固定长度的输出,这个输出通常被称为哈希值或...

    随记app,微博与博客整合

    总的来说,随记App是一个涵盖Android客户端开发和Java后端服务的综合项目,涉及到移动应用开发的多个核心领域,包括用户界面设计、网络编程、数据存储、用户认证以及服务器端的业务逻辑处理。这个项目对于学习和理解...

    EHCache的使用随记

    **EHCache的使用随记** EHCache是一款广泛应用于Java环境中的高效、易用且功能丰富的内存缓存系统。它能够显著提升应用性能,通过将常用数据存储在内存中,避免了反复从数据库读取,降低了I/O延迟。本文将探讨...

    2021-2022年收藏的精品资料辛辛那提1000维修随记.doc

    【辛辛那提1000维修随记】是一份关于数控系统维修的珍贵文档,主要涉及美国辛辛那提·米拉克龙公司的ACRAMATIC系列数控装置,特别是1994年推出的先进CNC控制系统——A2100。这篇随记详细记录了从1950年代早期的数控...

    2015年8月整理笔记本随记.ppt

    这篇随记内容涵盖了多个教育和德育相关的知识点,深入探讨了教育的本质、电子智慧教育的定义、学生文化的理解以及班主任的角色。以下是对这些知识点的详细解释: 1. **教育理念**: - "精准,精减,精彩":这强调...

    HL_Letter:个人随记

    9. **设计模式**:Java中常见的设计模式如单例、工厂、观察者、装饰者、代理等,是提高代码可读性和可维护性的有效工具。 10. **JVM内存模型**:理解Java虚拟机的堆、栈、方法区等内存区域对于优化程序性能至关重要...

    c++随记.docx

    ### C++ 随记知识点总结 #### 一、内存管理与 new/delete 操作符 在 C++ 中,`new` 和 `delete` 是用于动态内存分配的关键字。`new` 用于在堆区分配内存,`delete` 用于释放之前通过 `new` 分配的内存。使用 `new`...

    随记:flex发送XML到servlet

    标题“随记:flex发送XML到servlet”指的是在Flex应用程序中向Java Servlet发送XML数据的过程,这通常涉及到客户端与服务器端的交互。Flex是一种基于Adobe AIR或Flash Player运行时的开发框架,常用于创建富互联网...

    随记_电气_

    【标题】:“随记_电气_”提示我们这是一份关于电气工程领域的个人笔记或学习心得,可能包含了一些作者在学习或实践中积累的电气知识。 【描述】:“电气相关的知识,随手写的,不知道行不行11111”表明这份文档...

    互联网公司实习日记随记参考.pdf

    互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf

    三天不读书&智商输给猪-世界读书日ppt随记PPT模板.pptx

    三天不读书&智商输给猪-世界读书日ppt随记PPT模板.pptx

Global site tag (gtag.js) - Google Analytics