标签:自己 字符 引用 习惯 class ide 多个 通过 rap
装饰器模式
装饰器模式又称为包装(Wrapper)模式。装饰器模式以多客户端透明的方式扩展对象的功能,是继承关系的一个替代方案。
装饰器模式的结构
通常给对象添加功能,要么直接修改对象添加相应的功能,要么派生子类来扩展,抑或是使用对象组合的方式。显然,直接修改对应的类的方式并不可取,在面向对象的设计中,我们应该尽量使用组合对象而不是继承对象来扩展和复用功能,装饰器模式就是基于对象组合的方式的。
装饰器模式以对客户端透明的方式动态地给一个对象附加上了更多的责任。换言之,客户端并不会角色对象在装饰前和装饰后有什么不同。装饰器模式可以在不用创建更多子类的情况下,将对象的功能加以扩展。
装饰器模式中的角色有:
1、抽象构件角色
给出一个抽象接口,以规范准备接受附加责任的对象
2、具体构件角色
定义一个将要接受附加责任的类
3、装饰角色
持有一个构建对象的实例,并定义一个与抽象构件接口一致的接口
4、具体装饰角色
负责给构建对象贴上附加的责任
装饰器模式的例子
现在有这么一个场景:
1、有一批厨师,简单点吧,就全是中国厨师,他们有一个共同的动作是做晚饭
2、这批厨师做晚饭前的习惯不同,有些人喜欢做晚饭前洗手、有些人喜欢做晚饭前洗头
那么,按照装饰器模式,先抽象出抽象构建角色,Cook接口:
public interface Cook { public void cookDinner(); }
具体构建角色,中国厨师:
public class ChineseCook implements Cook { @Override public void cookDinner() { System.out.println("中国人做晚饭"); } }
定义一个装饰器角色,具体的工作具体装饰器去实现,这样,比如美国厨师做晚饭前也先洗手或者先洗头,这两个动作就可以做到复用,装饰器角色定义为FilterCook,很简单,实现Cook接口并持有Cook的引用:
public abstract class FilterCook implements Cook { protected Cook cook; }
最后定义一个具体装饰角色,该洗手的洗手,该洗头的洗头:
public class WashHandsCook extends FilterCook { public WashHandsCook(Cook cook) { this.cook = cook; } @Override public void cookDinner() { System.out.println("先洗手"); cook.cookDinner(); } }
public class WashHearCook extends FilterCook { public WashHearCook(Cook cook) { this.cook = cook; } @Override public void cookDinner() { System.out.println("先洗头"); cook.cookDinner(); } }
调用方这么实现:
@Test public void testDecorate() { Cook cook0 = new WashHandsCook(new ChineseCook()); Cook cook1 = new WashHearCook(new ChineseCook()); cook0.cookDinner(); cook1.cookDinner(); }
运行结果为:
先洗手 中国人做饭 先洗头 中国人做饭
简单的一个例子,实现了装饰器模式的两个功能点:
这就是装饰器模式。
装饰器模式与Java字节输入流InputStream
上面的例子可能写得不是很清楚,因此这里再继续用代码示例讲解装饰器模式。
装饰器模式在Java体系中的经典应用是Java I/O,下面先讲解字节输入流InputStream,再讲解字符输入流Reader,希望可以通过这两种输入流的讲解,加深对于装饰器模式的理解。
首先看一下字节输入流InputStream的类结构体系:
InputStream是一个顶层的接口,文章开头就说,装饰器模式是继承关系的一种替代方案,看一下为什么:
这样就导致两个问题:
所以,这个的时候我们就想到了一种解决方案:
这就是装饰器模式简单的由来,一切都是为了解决实际问题而诞生。下一步,根据UML图,我们来划分一下装饰器模式的角色。
1、InputStream是一个抽象构件角色:
public abstract class InputStream implements Closeable { // SKIP_BUFFER_SIZE is used to determine the size of skipBuffer private static final int SKIP_BUFFER_SIZE = 2048; // skipBuffer is initialized in skip(long), if needed. private static byte[] skipBuffer; ... }
2、ByteArrayInputStream、FileInputStream、ObjectInputStream、PipedInputStream都是具体构建角色,比如FileInputStream,它的声明是:
public class FileInputStream extends InputStream { /* File Descriptor - handle to the open file */ private FileDescriptor fd; private FileChannel channel = null; ... }
3、FilterInputStream无疑就是一个装饰角色,因为FilterInputStream实现了InputStream内的所有抽象方法并且持有一个InputStream的引用:
public class FilterInputStream extends InputStream { /** * The input stream to be filtered. */ protected volatile InputStream in; ... }
4、具体装饰角色就是InflaterInputStream、BufferedInputStream、DataInputStream,比如BufferedInputStream的声明就是:
public class BufferedInputStream extends FilterInputStream { private static int defaultBufferSize = 8192; /** * The internal buffer array where the data is stored. When necessary, * it may be replaced by another array of * a different size. */ protected volatile byte buf[]; ... }
搞清楚具体角色之后,我们就可以这么写了:
public static void main(String[] args) throws Exception { File file = new File("D:/aaa.txt"); InputStream in0 = new FileInputStream(file); InputStream in1 = new BufferedInputStream(new FileInputStream(file)); InputStream in2 = new DataInputStream(new BufferedInputStream(new FileInputStream(file))); }
我们这里实例化出了三个InputStream的实现类:
同理,我要给ByteArrayInputStream、ObjectInputStream增加功能,也可以使用类似的方法,整个过程中,最重要的是要理解几个问题:
装饰器模式与Java字符输入流Reader
看完了上面的解读,相信大家对于装饰器模式应当有了一定的理解,那么再来看一下Java字符输入流Reader,来加深对于装饰器模式的印象。
简单看一下Reader的类体系结构:
根据UML,分析一下每个角色:
1、抽象构建角色
毫无疑问,由Reader来扮演,它是一个抽象类,没有具体功能
2、具体构建角色
由InputStreamReader、CharArrayReader、PipedReader、StringReader来扮演
3、装饰角色
由FilterReader来扮演,但是这里要提一下这个BufferedReader,它本身也可以作为装饰角色出现,看一下BufferedReader的继承关系:
public class BufferedReader extends Reader { private Reader in; private char cb[]; private int nChars, nextChar; private static final int INVALIDATED = -2; private static final int UNMARKED = -1; private int markedChar = UNMARKED; ... }
看到BufferedReader是Reader的子类,且持有Reader的引用,因此这里的BufferedReader是可以被认为是一个装饰角色的。
4、具体装饰角色
BufferedReader上面提到了扮演了装饰角色,但是也可以被认为是一个具体装饰角色。除了BufferedReader,具体装饰角色还有PushbackReader。FileReader尽管也在第三行,但是FileReader构不成一个具体装饰角色,因为它不是BufferedReader的子类也不是FilterReader的子类,不持有Reader的引用。
半透明装饰器模式与全透明装饰器模式
再说一下半透明装饰器模式与全透明装饰器模式,它们的区别是:
全透明装饰器模式是一种比较理想主义的想法,现实中不太可能出现。
比如BufferedInputStream吧,我把FileInputStream装饰为BufferedInputStream,难道BufferedInputStream就完全没有自己的行为?比如返回缓冲区的大小、清空缓冲区(这里只是举个例子,实际BufferedInputStream是没有这两个动作的),这些都是InputStream本身不具备的,因为InputStream根本不知道缓冲区这个概念,它只知道定义读数据相关方法。
所以,更多的我们是采用半透明的装饰器模式,即允许装饰后的类中有属于自己的方法,因此,前面的I/O代码示例可以这么改动:
public static void main(String[] args) throws Exception { File file = new File("D:/aaa.txt"); FileInputStream in0 = new FileInputStream(file); BufferedInputStream in1 = new BufferedInputStream(new FileInputStream(file)); DataInputStream in2 = new DataInputStream(new BufferedInputStream(new FileInputStream(file))); }
这样才更有现实意义。
装饰器模式的优缺点
优点
1、装饰器模式与继承关系的目的都是要扩展对象的功能,但是装饰器模式可以提供比继承更多的灵活性。装饰器模式允许系统动态决定贴上一个需要的装饰,或者除掉一个不需要的装饰。继承关系是不同,继承关系是静态的,它在系统运行前就决定了
2、通过使用不同的具体装饰器以及这些装饰类的排列组合,设计师可以创造出很多不同的行为组合
缺点
由于使用装饰器模式,可以比使用继承关系需要较少数目的类。使用较少的类,当然使设计比较易于进行。但是另一方面,由于使用装饰器模式会产生比使用继承关系更多的对象,更多的对象会使得查错变得困难,特别是这些对象看上去都很像。
装饰器模式和适配器模式的区别
其实适配器模式也是一种包装(Wrapper)模式,它们看似都是起到包装一个类或对象的作用,但是它们使用的目的非常不一样:
1、适配器模式的意义是要将一个接口转变成另外一个接口,它的目的是通过改变接口来达到重复使用的目的
2、装饰器模式不要改变被装饰对象的接口,而是恰恰要保持原有的借口哦,但是增强原有接口的功能,或者改变元有对象的处理方法而提升性能
所以这两种设计模式的目的是不同的。
标签:自己 字符 引用 习惯 class ide 多个 通过 rap
原文地址:https://www.cnblogs.com/h-c-g/p/10690071.html