《Java设计模式》之装饰模式
Posted gavanwanggw
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了《Java设计模式》之装饰模式相关的知识,希望对你有一定的参考价值。
装饰模式(Decorator)
1. 装饰模式(Decorator)的定义:又名包装(Wrapper)模式。装饰模式以对client透明的方式扩展对象的功能,是继承关系的一个替代方案。
2. 装饰模式以对client透明的方式动态的给一个对象附加上很多其它的责任。换言之client并不会觉的对象在装饰前和装饰后有什么差别。
3. 装饰模式能够在不创造很多其它的子类的模式下,将对象的功能加以扩展。
4. 装饰模式与类继承的差别:
1) 装饰模式是一种动态行为,对已经存在类进行任意组合,而类的继承是一种静态的行为。一个类定义成什么样的,该类的对象便具有什么样的功能。无法动态的改变。
2) 装饰模式扩展的是对象的功能。不须要添加类的数量。而类继承扩展是类的功能。在继承的关系中,假设我们想添加一个对象的功能。我们仅仅能通过继承关系,在子类中添加两个方法。
3) 装饰与继承比較图:
4) 装饰模式是在不改变原类文件和使用继承的情况下,动态的扩展一个对象的功能,它是通过创建一个包装对象。也就是装饰来包裹真是的对象。
5. 装饰模式把对client的调用委派给被装饰的类,装饰模式的关键在于这样的扩展全然透明的。
6. 装饰模式的构成:
1) 抽象构建角色(Component):给出一个抽象的接口,以规范准备接受附加责任的对象。
相当于i/o流里面InputStream/OutputStream和Reader/Writer。
2) 详细的构建角色(ConcreteComponent):定义一个将要接受附加责任的类。
相当于i/o里面的FileOutputStream和FileInputStream。
3) 装饰角色(Docorator):持有一个抽象构建(Component)角色的引用。并定义一个与抽象构件一致的接口。
相当于i/o里面的FilerOutputStream和FilterInputStream。
4) 详细的装饰角色(ConcreteDecorator):负责给构建对象“贴上”附加的责任。
相当于i/o流里面的BufferedOutputStream和BufferedInputStream以及DataOutputStream和DataInputSrtream。
7. 装饰模式的特点:
1) 装饰对象和真实对象具有同样的接口。这样client对象就能够以真实对象的同样的方式和装饰对象交互。
2) 装饰对象包括一个真实对象的引用(reference).
3) 装饰对象接受全部来自client的请求,它把这些请求转发给真实的对象。
4) 装饰对象能够在转发这些请求曾经或者以后添加一些附加的功能。这样就能确保在执行时,不用改动给定对象结构就能够在外部添加附加的功能。在面向对象的程序设计中,一般是使用继承的关系来扩展给定类的功能。
装饰模式的结构
装饰模式以对客户透明的方式动态地给一个对象附加上很多其它的责任。换言之。client并不会认为对象在装饰前和装饰后有什么不同。装饰模式能够在不使用创造很多其它子类的情况下,将对象的功能加以扩展。
装饰模式的类图例如以下:
在装饰模式中的角色有:
● 抽象构件(Component)角色:给出一个抽象接口,以规范准备接收附加责任的对象。
● 详细构件(ConcreteComponent)角色:定义一个将要接收附加责任的类。
● 装饰(Decorator)角色:持有一个构件(Component)对象的实例,并定义一个与抽象构件接口一致的接口。
● 详细装饰(ConcreteDecorator)角色:负责给构件对象“贴上”附加的责任。
源码
抽象构件角色
- package com.bankht.Decorator;
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:21:22
- *
- * @类说明 :抽象构件角色
- */
- public interface Component {
- public void sampleOperation();
- }
详细构件角色
- package com.bankht.Decorator;
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:22:05
- *
- * @类说明 :详细构件角色
- */
- public class ConcreteComponent implements Component {
- @Override
- public void sampleOperation() {
- // TODO Auto-generated method stub
- // 写相关的业务代码
- }
- }
装饰角色
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:23:03
- *
- * @类说明 :装饰角色
- */
- public class Decorator implements Component {
- private Component component;
- public Decorator(Component component) {
- this.component = component;
- }
- @Override
- public void sampleOperation() {
- // 委派给构件
- component.sampleOperation();
- }
- }
详细装饰角色
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:23:34
- *
- * @类说明 :详细装饰角色
- */
- public class ConcreteDecoratorA extends Decorator {
- public ConcreteDecoratorA(Component component) {
- super(component);
- }
- @Override
- public void sampleOperation() {
- super.sampleOperation();
- // 写相关的业务代码
- }
- }
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:23:34
- *
- * @类说明 :详细装饰角色
- */
- public class ConcreteDecoratorB extends Decorator {
- public ConcreteDecoratorB(Component component) {
- super(component);
- }
- @Override
- public void sampleOperation() {
- super.sampleOperation();
- // 写相关的业务代码
- }
- }
齐天大圣的样例
孙悟空有七十二般变化,他的每一种变化都给他带来一种附加的本领。他变成鱼儿时。就能够到水里游泳;他变成鸟儿时,就能够在天上飞行。
本例中,Component的角色便由鼎鼎大名的齐天大圣扮演;ConcreteComponent的角色属于大圣的本尊。就是猢狲本人;Decorator的角色由大圣的七十二变扮演。而ConcreteDecorator的角色便是鱼儿、鸟儿等七十二般变化。
源码
抽象构件角色“齐天大圣”接口定义了一个move()方法,这是全部的详细构件类和装饰类必须实现的。
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:28:23
- *
- * @类说明 :抽象构件角色“齐天大圣”接口定义了一个move()方法,这是全部的详细构件类和装饰类必须实现的。
- */
- // 大圣的尊号
- public interface TheGreatestSage {
- public void move();
- }
详细构件角色“大圣本尊”猢狲类
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:28:50
- *
- * @类说明 :详细构件角色“大圣本尊”猢狲类
- */
- public class Monkey implements TheGreatestSage {
- @Override
- public void move() {
- // 代码
- System.out.println("Monkey Move");
- }
- }
抽象装饰角色“七十二变”
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:29:24
- *
- * @类说明 :抽象装饰角色“七十二变”
- */
- public class Change implements TheGreatestSage {
- private TheGreatestSage sage;
- public Change(TheGreatestSage sage) {
- this.sage = sage;
- }
- @Override
- public void move() {
- // 代码
- sage.move();
- }
- }
详细装饰角色“鱼儿”
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:29:47
- *
- * @类说明 :详细装饰角色“鱼儿”
- */
- public class Fish extends Change {
- public Fish(TheGreatestSage sage) {
- super(sage);
- }
- @Override
- public void move() {
- // 代码
- System.out.println("Fish Move");
- }
- }
详细装饰角色“鸟儿”
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:30:11
- *
- * @类说明 :详细装饰角色“鸟儿”
- */
- public class Bird extends Change {
- public Bird(TheGreatestSage sage) {
- super(sage);
- }
- @Override
- public void move() {
- // 代码
- System.out.println("Bird Move");
- }
- }
client类
- /**
- * @author: 特种兵—AK47
- * @创建时间:2012-6-26 上午09:30:32
- *
- * @类说明 :client类
- */
- public class Client {
- public static void main(String[] args) {
- TheGreatestSage sage = new Monkey();
- // 第一种写法
- TheGreatestSage bird = new Bird(sage);
- TheGreatestSage fish = new Fish(bird);
- // 另外一种写法
- // TheGreatestSage fish = new Fish(new Bird(sage));
- fish.move();
- bird.move();
- }
- }
“大圣本尊”是ConcreteComponent类,而“鸟儿”、“鱼儿”是装饰类。
要装饰的是“大圣本尊”,也即“猢狲”实例。
上面的样例中,系统把大圣从一仅仅猢狲装饰成了一仅仅鸟儿(把鸟儿的功能加到了猢狲身上),然后又把鸟儿装饰成了一条鱼儿(把鱼儿的功能加到了猢狲+鸟儿身上,得到了猢狲+鸟儿+鱼儿)。
如上图所看到的,大圣的变化首先将鸟儿的功能附加到了猢狲身上。然后又将鱼儿的功能附加到猢狲+鸟儿身上。
装饰模式的简化
大多数情况下。装饰模式的实现都要比上面给出的示意性样例要简单。
假设仅仅有一个ConcreteComponent类,那么能够考虑去掉抽象的Component类(接口),把Decorator作为一个ConcreteComponent子类。例如以下图所看到的:
假设仅仅有一个ConcreteDecorator类。那么就没有必要建立一个单独的Decorator类,而能够把Decorator和ConcreteDecorator的责任合并成一个类。甚至在仅仅有两个ConcreteDecorator类的情况下,都能够这样做。例如以下图所看到的:
透明性的要求
装饰模式对client的透明性要求程序不要声明一个ConcreteComponent类型的变量,而应当声明一个Component类型的变量。
用孙悟空的样例来说,必须永远把孙悟空的全部变化都当成孙悟空来对待。而假设把老孙变成的鱼儿当成鱼儿,而不是老孙。那就被老孙骗了,而这时不应当发生的。
以下的做法是对的:
- TheGreatestSage bird = new Bird(sage);
而以下的做法是不正确的:
- Bird bird = new Bird(sage);
半透明的装饰模式
然而。纯粹的装饰模式非常难找到。装饰模式的用意是在不改变接口的前提下,增强所考虑的类的性能。在增强性能的时候。往往须要建立新的公开的方法。即便是在孙大圣的系统里,也须要新的方法。比方齐天大圣类并没有飞行的能力。而鸟儿有。这就意味着鸟儿应当有一个新的fly()方法。
再比方,齐天大圣类并没有游泳的能力。而鱼儿有,这就意味着在鱼儿类里应当有一个新的swim()方法。
这就导致了大多数的装饰模式的实现都是“半透明”的,而不是全然透明的。换言之,同意装饰模式改变接口,添加新的方法。这意味着client能够声明ConcreteDecorator类型的变量,从而能够调用ConcreteDecorator类中才有的方法:
- Bird bird = new Bird(sage);
- bird.fly();
半透明的装饰模式是介于装饰模式和适配器模式之间的。适配器模式的用意是改变所考虑的类的接口,也能够通过改写一个或几个方法,或添加新的方法来增强或改变所考虑的类的功能。大多数的装饰模式实际上是半透明的装饰模式,这种装饰模式也称做半装饰、半适配器模式。
装饰模式的长处
(1)装饰模式与继承关系的目的都是要扩展对象的功能,可是装饰模式能够提供比继承很多其它的灵活性。装饰模式同意系统动态决定“贴上”一个须要的“装饰”,或者除掉一个不须要的“装饰”。
继承关系则不同,继承关系是静态的,它在系统执行前就决定了。
(2)通过使用不同的详细装饰类以及这些装饰类的排列组合。设计师能够创造出非常多不同行为的组合。
装饰模式的缺点
因为使用装饰模式。能够比使用继承关系须要较少数目的类。
使用较少的类,当然使设计比較易于进行。可是,在还有一方面,使用装饰模式会产生比使用继承关系很多其它的对象。很多其它的对象会使得查错变得困难,特别是这些对象看上去都非常相像。
设计模式在JAVA I/O库中的应用
装饰模式在Java语言中的最著名的应用莫过于Java I/O标准库的设计了。
因为Java I/O库须要非常多性能的各种组合。假设这些性能都是用继承的方法实现的,那么每一种组合都须要一个类。这样就会造成大量性能反复的类出现。而假设採用装饰模式。那么类的数目就会大大降低。性能的反复也能够减至最少。因此装饰模式是Java I/O库的基本模式。
Java I/O库的对象结构图例如以下。因为Java I/O的对象众多,因此仅仅画出InputStream的部分。
依据上图能够看出:
● 抽象构件(Component)角色:由InputStream扮演。这是一个抽象类,为各种子类型提供统一的接口。
● 详细构件(ConcreteComponent)角色:由ByteArrayInputStream、FileInputStream、PipedInputStream、StringBufferInputStream等类扮演。它们实现了抽象构件角色所规定的接口。
● 抽象装饰(Decorator)角色:由FilterInputStream扮演。
它实现了InputStream所规定的接口。
● 详细装饰(ConcreteDecorator)角色:由几个类扮演,各自是BufferedInputStream、DataInputStream以及两个不经常使用到的类LineNumberInputStream、PushbackInputStream。
半透明的装饰模式
装饰模式和适配器模式都是“包装模式(Wrapper Pattern)”,它们都是通过封装其它对象达到设计的目的的,可是它们的形态有非常大差别。
理想的装饰模式在对被装饰对象进行功能增强的同一时候。要求详细构件角色、装饰角色的接口与抽象构件角色的接口全然一致。而适配器模式则不然,一般而言,适配器模式并不要求对源对象的功能进行增强,可是会改变源对象的接口,以便和目标接口相符合。
装饰模式有透明和半透明两种。这两种的差别就在于装饰角色的接口与抽象构件角色的接口是否全然一致。透明的装饰模式也就是理想的装饰模式。要求详细构件角色、装饰角色的接口与抽象构件角色的接口全然一致。相反,假设装饰角色的接口与抽象构件角色接口不一致,也就是说装饰角色的接口比抽象构件角色的接口宽的话,装饰角色实际上已经成了一个适配器角色,这样的装饰模式也是能够接受的,称为“半透明”的装饰模式。例如以下图所看到的。
在适配器模式里面,适配器类的接口一般会与目标类的接口重叠,但往往并不全然同样。
换言之。适配器类的接口会比被装饰的目标类接口宽。
显然,半透明的装饰模式实际上就是处于适配器模式与装饰模式之间的灰色地带。假设将装饰模式与适配器模式合并成为一个“包装模式”的话。那么半透明的装饰模式倒能够成为这样的合并后的“包装模式”的代表。
InputStream类型中的装饰模式
InputStream类型中的装饰模式是半透明的。为了说明这一点,最好还是看一看作装饰模式的抽象构件角色的InputStream的源码。这个抽象类声明了九个方法,并给出了当中八个的实现,另外一个是抽象方法。须要由子类实现。
- public abstract int read() throws IOException;
- public int read(byte b[]) throws IOException {}
- public int read(byte b[], int off, int len) throws IOException {}
- public long skip(long n) throws IOException {}
- public int available() throws IOException {}
- public void close() throws IOException {}
- public synchronized void mark(int readlimit) {}
- public synchronized void reset() throws IOException {}
- public boolean markSupported() {}
- }
以下是作为装饰模式的抽象装饰角色FilterInputStream类的源码。能够看出,FilterInputStream的接口与InputStream的接口是全然一致的。也就是说,直到这一步。还是与装饰模式相符合的。
- protected FilterInputStream(InputStream in) {}
- public int read() throws IOException {}
- public int read(byte b[]) throws IOException {}
- public int read(byte b[], int off, int len) throws IOException {}
- public long skip(long n) throws IOException {}
- public int available() throws IOException {}
- public void close() throws IOException {}
- public synchronized void mark(int readlimit) {}
- public synchronized void reset() throws IOException {}
- public boolean markSupported() {}
- }
以下是详细装饰角色PushbackInputStream的源码。
- private void ensureOpen() throws IOException {}
- public PushbackInputStream(InputStream in, int size) {}
- public PushbackInputStream(InputStream in) {}
- public int read() throws IOException {}
- public int read(byte[] b, int off, int len) throws IOException {}
- public void unread(int b) throws IOException {}
- public void unread(byte[] b, int off, int len) throws IOException {}
- public void unread(byte[] b) throws IOException {}
- public int available() throws IOException {}
- public long skip(long n) throws IOException {}
- public boolean markSupported() {}
- public synchronized void mark(int readlimit) {}
- public synchronized void reset() throws IOException {}
- public synchronized void close() throws IOException {}
- }
查看源代码,你会发现,这个装饰类提供了额外的方法unread(),这就意味着PushbackInputStream是一个半透明的装饰类。
换言 之。它破坏了理想的装饰模式的要求。假设client持有一个类型为InputStream对象的引用in的话,那么假设in的真实类型是 PushbackInputStream的话,仅仅要client不须要使用unread()方法,那么client一般没有问题。可是假设client必须使用这种方法,就 必须进行向下类型转换。
将in的类型转换成为PushbackInputStream之后才可能调用这种方法。可是,这个类型转换意味着client必须知道它 拿到的引用是指向一个类型为PushbackInputStream的对象。这就破坏了使用装饰模式的原始用意。
现实世界与理论总归是有一段差距的。纯粹的装饰模式在真实的系统中非常难找到。一般所遇到的,都是这样的半透明的装饰模式。
以下是使用I/O流读取文件内容的简单操作演示样例。
- public static void main(String[] args) throws IOException {
- // 流式读取文件
- DataInputStream dis = null;
- try{
- dis = new DataInputStream(
- new BufferedInputStream(
- new FileInputStream("test.txt")
- )
- );
- //读取文件内容
- byte[] bs = new byte[dis.available()];
- dis.read(bs);
- String content = new String(bs);
- System.out.println(content);
- }finally{
- dis.close();
- }
- }
- }