`
kakajw
  • 浏览: 265101 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

从Decorator,Adapter模式看Java的IO库

阅读更多

讲到Decorator模式的时候,不能不提到它的实际应用--在Java/IO库里面的应用,<<Java与模式>>这本书也不例外,有点不一样的是,这本书在介绍的时候有个专题,是从两个模式来看Java/IO库,完这个专题后,个人感觉对Java/IO库有了全新的认识同时也加深了Decorator模式跟Adapter适配器模式的理解,现和大家分享下这个在我看来很伟大的成果,同时说明下,以下大部分文字跟图片是来自<<Java与模式>>这本书。

 

 一.引子(概括地介绍Java的IO)

 无论是哪种编程语言,输入跟输出都是重要的一部分,Java也不例外,而且Java将输入/输出的功能和使用范畴做了很大的扩充。它采用了流的机制来实现输入/输出,所谓流,就是数据的有序排列,而流可以是从某个源(称为流源或Source of Stream)出来,到某个目的地(称为流汇或Sink of Stream)去的。由流的方向,可以分成输入流和输出流,一个程序从输入流读取数据向输出流写数据。

 如,一个程序可以用FileInputStream类从一个磁盘文件读取数据,如下图所示:


 像FileInputStream这样的处理器叫做流处理器,它就像流的管道一样,从一个流源吸入某种类型的数据,并输出某种类型的数据。上面这种示意图叫做流的管道图。

 同样道理,也可以用FileOutputStream类向一个磁盘文件写数据,如下图所示:


 

 在实际应用这种机制并不没有太大的用处,程序需要写出地通常是非常结构化的信息,因此这些byte类型的数据实际上是一些数值,文字,源代码等。Java的I/O库提供了一个称做链接(Chaining)的机制,可以将一个流处理器跟另一个流处理器首尾相接,以其中之一的输出为输入,形成一个流管道的链接。

 例如,DataInputStream流处理器可以把FileInputStream流对象的输出当作输入,将Byte类型的数据转换成Java的原始类型和String类型的数据。如下图所示:


 

 类似地,向一个文件写入Byte类型的数据不是一个简单的过程。一个程序需要向一个文件里写入的数据往往都是结构化的,而Byte类型则是原始类型。因此在写的时候必须经过转换。DataOutputStream流处理器提供了接收了原始数据类型和String数据类型,而这个流处理器的输出数据则是Byte类型。也就是说DataOutputStream可以将源数据转换成Byte类型的数据,再输出来。

 这样一来,就可以将DataOutputStream与FileOutputStream链接起来,这样程序就可以将原始数据类型和String类型的源数据写入这个链接好的双重管道里面,达到将结构化数据写到磁盘文件里面的目的,如下图所示:


 

 这又是链接的所发挥的大作用。

 流处理器所处理的流必定都有流源,而如果将流类所处理的流源分类的话,基本可以分成两大类:

 第一 数组,String,File等,这一种叫原始流源。

 第二 同样类型的流用做链接流类的流源,叫链接流源。

 

二 Java I/O库的设计原则

 Java语言的I/O库是对各种常见的流源,流汇以及处理过程的抽象化。客户端的Java程序不必知道最终的流源,流汇是磁盘上的文件还是数组等;也不必关心数据是否经过缓冲的,可否按照行号读取等处理的细节。

 书中提到了,对于第一次见到Java/IO库的人,无不因为这个库的庞杂而感到困惑;而对于熟悉这个库的人,而又常常为这个库的设计是否得当而争论不体。书的作者提出自己的意见,要理解Java I/O这个庞大而复杂的库,关键是要掌握两个对称性跟两个设计模式模式。

 

Java I/O库具有两个对称性,它们分别是:

 1 输入-输出对称性,比如InputStream和OutputStream各自占据Byte流的输入与输出的两个平行的等级结构的根部。而Reader和Writer各自占据Char流的输入与输出的两个平行的等级结构的根部。

 2 byte-char对称,InputStream和Reader的子类分别负责Byte和Char流的输入;OutputStream和Writer的子类分别负责Byte和Char流的输出,它们分别形成平行的等级结构。

 

Java I/O库的两个设计模式:

Java的I/O库总体设计是符合装饰者模式(Decorator)跟适配器模式(Adapter)的。如前所述,这个库中处理流的类叫做流类。引子里所谈到的FileInputStream,FileOutputStream,DataInputStream及DataOutputStream都是流处理器的例子。

 

1 装饰者模式:在由InputStream,OutputStream,Reader和Writer代表的等级结构内部,有一些流处理器可以对另一些流处理器起到装饰作用,形成新的,具有改善了的功能的流处理器。装饰者模式是Java I/O库的整体设计模式。这样的一个原则是符合装饰者模式的,如下图所示:


2 适配器模式:在由InputStream,OutputStream,Reader和Writer代表的等级结构内部,有一些流处理器是对其它类型的流源的适配。这就是适配器模式的应用,如下图所示。

  

 适配器模式应用到了原始流处理器的设计上面,构成了I/O库所有流处理器的起点。

 

 

三 装饰模式的应用

 学过装饰模式后,大家会发现,它在Java语言中最著名的应用莫过于Java I/O标准为库的设计了。这一节将以处理Byte流为例,看看装饰模式是怎样得到应用的。

 

 为什么不用继承而用装饰模式

 我们知道Java I/O库需要很多性能的各种组合,如果说这些性能的组合是通过继承方式来实现的话,那么每一种组合都需要一个类,这样就会出现大量重复性问题的出现,从而使类数目“爆炸”。而如果采用装饰模式,那么不仅类的数目大减少了,性能的重复也可以减至到最少。所以装饰模式是Java I/O库的基本模式。在这里我想再用<<Head First Design Pattern>>中讲到装饰模式时候的一个例子,看看装饰模式是怎么达到不仅类的数目大减少了,性能的重复也可以减至到最少。

 再回到Java I/O库,由于装饰模式的引用,造成了灵活性和复杂都大大增加了,我们在使用Java I/O库时,必须理解Java I/O库是由一些基本的原始流处理器和围绕它们的装饰流处理器所组成的,这样可以在学习和使用Java I/O库时达到事半功倍的效果。

 下面我用<<Java与模式>>,<<Head First Design Pattern>>或者是网上看到的一些类图来分析:

 

 首先是InputStream类型中的装饰模式:

 InputStream有七个直接的具体子类,有四个属于FilterInputStream的具体子类,如下图所示:

  上图中所有的类都叫做流处理器,这个图就叫做(InputStream类型的)流处理器图。

  书中提到根据输入流的源的类型,可以将这些流类分成两种,即原始流类(Original Stream)和链接流处理器(Wrapper Stream)。

 

  原始流处理器

  原始流处理器接收一个Byte数组对象,String对象,FileDiscriptor对象或者不同类型的流源对象,根据上面的图,原始流处理器包括以下四种:

  ByteArrayInputStream:为多线程的通信提供缓冲区操作功能,接收一个Byte数组作为流的源。

  FileInputStream:建立一个与文件有关的输入流。接收一个File对象作为流的源。

  PipedInputStream:可以与PipedOutputStream配合使用,用于读入一个数据管道的数据,接收一个PipedOutputStream作为源。

  StringBufferInputStream:将一个字符串缓冲区转换为一个输入流。接收一个String对象作为流的源。(JDK帮助文档上说明:已过时。此类未能正确地将字符转换为字节。从JDK1.1开始,从字符串创建流的首选方法是通过StringReader类进行创建。只有字符串中每个字符的低八位可以由此类使用。)

 

链接流处理器

  所谓链接流处理器,就是可以接收另一个流对象作为源,并对之进行功能扩展的类。InputStream类型的链接处理器包括以下几种,它们都接收另一个InputStream对象作为流源。

  (1)FilterInputStream称为过滤输入流,它将另一个输入流作为流源。这个类的子类包括以下几种:

  BufferedInputStream:用来从硬盘将数据读入到一个内存缓冲区中,并从缓冲区提供数据。

  DataInputStream:提供基于多字节的读取方法,可以读取原始类型的数据。

  LineNumberInputStream:提供带有行计数功能的过滤输入流。

  PushbackInputStream:提供特殊的功能,可以将已经读取的字节“推回”到输入流中。

  (2)ObjectInputStream可以将使用ObjectInputStream串行化的原始数据类型和对象重新并行化。

  (3)SeqcueneInputStream可以将两个已有的输入流连接起来,形成一个输入流,从而将多个输入流排列构成一个输入流序列。

  抽象结构图

  按照上面的这种原始流处理器和链接流处理器的划分,可以用下面的结构图来描述它们之间的关系。

  

 

 上面的流处理器图跟装饰模式的结构图有着显而易见的相同之处。实际上InputStream类型的流处理器结构确实符合装饰模式。 

 装饰模式结构图

 

  对于上图FilterInputStream查看JDK1.6源代码,部分代码如下:

public class FilterInputStream extends InputStream {
    protected volatile InputStream in; 

    protected FilterInputStream(InputStream in) {
       this.in = in;
    }
    //其它代码
}

 

FilterInputStream继承了InputStream,也引用了InputStream,而它有四个子类,这就是所谓的Decorator模式。
  上面这个图向我们传达了这个信息:链接流对象接收一个原始流对象或者另外一个链接流对象作为流源;另一方面他们对流源的内部工作方法做了相应的改变,这种改变是装饰模式所要达到的目的。比如:

  BufferedInputStream“装饰”了InputStream的内部工作方式,使得流的读入操作使用了缓冲机制。在使用了缓冲机制后,不会对每一次的流读入操作都产生一个物理的读盘动作,从而提高了程序的效率,在汲及到物理流的读入时,都应当使用这个装饰流类。

  LineNumberInputStream和PushbackInputStream也同样“装饰”了InputStream的内部工作方式,前者使得程序能够按照行号读入数据;后者能够使程序读入的过程中,退后一个字符。

  DataInputStream子类读入各种不同的原始数据类型以及String类型的数据,这一点可以从它提供的各种read方法看出来,如:readByte(),readInt(),readFloat()等。

  Java语言的I/O库提供了四大等级结构:InputStream,OutputStream,Reader,Writer四个系列的类。InputStream和OutputStream处理8位字节流数据, Reader和Writer处理16位的字符流数据。InputStream和Reader处理输入, OutputStream和Writer处理输出,所以OutputStream,Reader,Writer这三类的装饰模式跟前面详细介绍的InputStream装饰模式大同小异,大家可以看书中其它部分对这三类的详细描述或者从网上也能找到有关资料。为了方便比较这几种类型,顺便附上Java语言的I/O层次结构图:

 下面的图表示:以InputStream和OutputStream形成的层次关系



 下面的图表示:以Reader和Writer形成的层次关系



 

 

四 适配器模式的应用

 适配器模式是Java I/O库中第二个最为重要的设计模式。

InputStream原始流处理器中的适配器模式,InputStream类型的原始流处理器是适配器模式的应用。

ByteArrayInputStream是一个适配器类
 ByteArrayInputStream继承了InputStream的接口,而封装了一个byte数组。换言之,它将一个byte数组的接口适配成InputStream流处理器的接口。
 我们知道Java语言支持四种类型:Java接口,Java类,Java数组,原始类型(即int,float等)。前三种是引用类型,类和数组的实例是对象,原始类型的值不是对象。
也即,Java语言的数组是像所有的其他对象一样的对象,而不管数组中所存储的元素类型是什么。
这样一来的话,ByteArrayInputStream就符合适配器模式的描述,是一个对象形式的适配器类。

 

FileInputStream是一个适配器类
在FileInputStream继承了InputStrem类型,同时持有一个对FileDiscriptor的引用。这是将一个FileDiscriptor对象适配成InputStrem类型的对象形式的适配器模式。

查看JDK1.6的源代码我们可以看到:

 

public  class FileInputStream extends InputStream
{
    private FileDescriptor fd;

    private FileChannel channel = null;

 public FileInputStream(File file) throws FileNotFoundException {
      String name = (file != null ? file.getPath() : null);
      SecurityManager security = System.getSecurityManager();
      if (security != null) {
        security.checkRead(name);
       }
        if (name == null) {
            throw new NullPointerException();
        }
       fd = new FileDescriptor();
       open(name);
    }

   //其它代码
}

 

StringBufferInputStream继承了InputStream类型,同时持有一个对String对象的引用,这是一个将String对象适配成InputStream类型的对象形式的适配器模式。

OutputStream原始流处理器中的适配器模式,同样地,在OutputStream类型中,所有的原始流处理器都是适配器类。

ByteArrayOutputStream继承了OutputStream类型,同时持有一个对byte数组的引用。它一个byte数组的接口适配成OutputString类型的接口,因此也是一个对象形式的适配器模式的应用。

 

FileOutputStream是一个适配器类
FileOutputStream继承了OutputStream类型,同时持有一个对FileDiscriptor对象的引用。这是一个将FileDiscriptor接口适配成OutputStream接口形式的对象形适配器模式。

Reader原始流处理器中的适配器模式,Reader类型的原始流处理器都是适配器模式的应用。

 

StringReader是一个适配器类
StringReader类继承了Reader类型,持有一个对String对象的引用。它将String的接口适 配成Reader类型的接口。

 

从byte流到char流的适配
在Java I/O库中,使用比较频繁的要数InputStreamReader,OutputStreamWriter这两种类了,
InputStreamReader是从byte输入流到char输入流的一个适配器。

当把InputStreamReader与任何InputStream的具体子类链接的时候,可以从InputStream的输出读入byte类型的数据,将之转换成为char类型的数据,如下图所示:


查看JDK1.6的InputStreamReader源代码:

 

public class InputStreamReader extends Reader {

    private final StreamDecoder sd;

    public InputStreamReader(InputStream in) {
       super(in);
        try {
           sd = StreamDecoder.forInputStreamReader(in, this, (String)null); 
          // ## check lock object
        } catch (UnsupportedEncodingException e) {
     // The default encoding should always be available
     throw new Error(e);
 }

 //其它代码
}

 

其中StreamDecoder是sun.nio.cs这个包里的一个类OutputStreamWriter是适配器类。

 35   public class StreamDecoder extends Reader
   36   {
   58       public static StreamDecoder forInputStreamReader(InputStream in,
   59                                                        Object lock,
   60                                                        String charsetName)
   61           throws UnsupportedEncodingException
   62       {
   63           String csn = charsetName;
   64           if (csn == null)
   65               csn = Charset.defaultCharset().name();
   66           try {
   67               if (Charset.isSupported(csn))
   68                   return new StreamDecoder(in, lock, Charset.forName(csn));
   69           } catch (IllegalCharsetNameException x) { }
   70           throw new UnsupportedEncodingException (csn);
   71       }

            // Exactly one of these is non-null
  225       private InputStream in;
  226       private ReadableByteChannel ch;
  227   
  228       StreamDecoder(InputStream in, Object lock, Charset cs) {
  229           this(in, lock,
  230            cs.newDecoder()
  231            .onMalformedInput(CodingErrorAction.REPLACE)
  232            .onUnmappableCharacter(CodingErrorAction.REPLACE));
  233       }
  234   
  235       StreamDecoder(InputStream in, Object lock, CharsetDecoder dec) {
  236           super(lock);
  237           this.cs = dec.charset();
  238           this.decoder = dec;
  239   
  240           // This path disabled until direct buffers are faster
  241           if (false && in instanceof FileInputStream) {
  242           ch = getChannel((FileInputStream)in);
  243           if (ch != null)
  244               bb = ByteBuffer.allocateDirect(DEFAULT_BYTE_BUFFER_SIZE);
  245           }
  246           if (ch == null) {
  247           this.in = in;
  248           this.ch = null;
  249           bb = ByteBuffer.allocate(DEFAULT_BYTE_BUFFER_SIZE);
  250           }
  251           bb.flip();                      // So that bb is initially empty
  252       }
              //其它代码

}

 

同样道理我们能得出OutputStringWriter是从OutputStream到Writer的适配器类。也就是说,与任何一个OutputStream的具体子类相链接时,OutputStringWriter可以将OutputStream类型的byte流适配成为char流。它的源代码跟上面的InputStreamReader差不多,这就不贴出来,感兴趣可以查看JDK1.4在线源码这本书后面还有个小例子,附有一些讲解,我就不列出来了,有书的可以看看。

 

五 总结

在这三篇文章里主要是汲及到三个知识点:知识点一: Java I/O库的四大等级结构
Java语言的I/O库提供了四大等级结构:InputStream,OutputStream,Reader,Writer四个系列的类。InputStream和OutputStream处理8位字节流数据, Reader和Writer处理16位的字符流数据。InputStream和Reader处理输入, OutputStream和Writer处理输出。
知识点二: Decorator模式在Java I/O库的应用
知识点三: Adapter模式在Java I/O库的应用

 

转自:http://blog.csdn.net/FG2006/archive/2011/05/09/6407325.aspx

  • 大小: 14.8 KB
  • 大小: 11 KB
  • 大小: 18.2 KB
  • 大小: 19.6 KB
  • 大小: 5.4 KB
  • 大小: 6.2 KB
  • 大小: 40.7 KB
  • 大小: 15.1 KB
  • 大小: 47.8 KB
  • 大小: 112.8 KB
  • 大小: 47.8 KB
  • 大小: 93.1 KB
  • 大小: 42.4 KB
  • 大小: 19.1 KB
分享到:
评论

相关推荐

    Java-从Decorator,Adapter模式看Java/IO

    Java的IO库是一个复杂而强大的系统,其设计思想和实现方式体现了多种设计模式,包括Decorator(装饰者)模式和Adapter(适配器)模式。本文将深入探讨这两个模式在Java/IO中的应用,并通过实例解释它们如何使得Java...

    JAVA设计模式

    Java的IO流类库大量使用了装饰器模式。 9. **桥接模式(Bridge)** 桥接模式将抽象部分与实现部分分离,使它们可以独立变化。它将抽象部分和实现部分解耦,使得它们可以独立变化和发展。 10. **代理模式(Proxy)...

    Java 设计模式最佳实践

    Java的IO流体系就是一个很好的装饰器模式应用示例。 六、策略模式(Strategy) 策略模式定义了一系列算法,并将每个算法封装起来,使它们可以相互替换。策略模式让算法的变化独立于使用它的客户。在Java中,可以...

    JAVA设计模式在JDK中的应用

    Java作为一门流行的编程语言,其标准库(JDK)中巧妙地融入了多种设计模式的应用。本文将详细介绍在JDK中体现的23种经典设计模式,并通过具体的代码示例进行说明。 #### 二、设计模式分类 设计模式可以大致分为三类...

    23种设计模式的Java实现

    Java中的`java.io.InputStream`和`java.io.OutputStream`的read和write方法就是模板方法模式的应用。 18. **访问者模式**(Visitor):行为型设计模式,表示一个作用于某对象结构中的各元素的操作。它使你可以在不...

    设计模式Java版各个实现代码

    Java中的IO流类库就大量使用了装饰器模式。 6. **适配器模式(Adapter Pattern)**:将一个类的接口转换成客户希望的另一个接口。适配器使原本由于接口不兼容而不能一起工作的那些类可以一起工作。在Java中,适配器...

    设计模式 in java

    Java中,这种模式常见于图形库,如Swing组件体系。 10. **策略模式(Strategy)**:定义一组算法,并将每一个算法封装起来,使它们可以互相替换。Java的策略模式常用于游戏AI、排序算法等场景。 11. **模板方法...

    设计模式与JAVA

    结构型模式则关注如何将对象和类组合成更大的结构,例如适配器模式(Adapter)允许不同接口的类协同工作,装饰器模式(Decorator)可以在运行时动态地为对象添加新的行为或职责,而代理模式(Proxy)可以为其他对象...

    设计模式_java语言中的应用

    在Java编程语言中,设计模式是一种经过验证的、在特定情境下解决软件设计问题的通用可复用解决方案。设计模式不仅能够提升代码的可读性和可维护性,还能提高开发效率,使得团队间的沟通更为顺畅。本文将深入探讨Java...

    设计模式(Java 与模式-笔记 二)

    Java的IO流系统大量使用了装饰器模式,例如`BufferedReader`和`InputStreamReader`。 此外,适配器模式(Adapter)是将两个不兼容的接口转换为可以互相操作的模式。在Java中,适配器模式常通过继承或组合实现,使得...

    探寻Java源码中的设计模式

    7. 装饰器模式(Decorator Pattern):装饰器模式动态地给一个对象添加额外的职责,如`java.io.InputStream`和`java.io.FilterInputStream`,后者可以在不改变原有行为的基础上增加新的功能。 8. 桥接模式(Bridge ...

    java设计模式实例

    在Java中,`java.io`包下的许多类如`BufferedReader`和`PrintWriter`就是装饰器模式的应用。 8. 组合模式(Composite): 组合模式将对象组织成树形结构,允许用户对单个对象和组合对象进行一致的处理。它在处理...

    设计模式java版.rar

    Java的IO流类如`BufferedReader`和`PrintWriter`都是装饰者模式的示例。 7. **代理模式(Proxy)**:为其他对象提供一种代理以控制对这个对象的访问。在Java中,可以使用JDK动态代理或者CGLIB等第三方库来实现。 8...

    JAVA设计模式.zip

    Java的IO流类库就是装饰器模式的经典应用。 八、适配器模式(Adapter) 适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。在Java中,我们可以使用适配器将旧的API转换成新的API接口,以适应新...

    设计模式java实现代码

    Java的IO流类库就大量使用了装饰器模式。 8. **适配器模式**(Adapter):将一个类的接口转换成客户期望的另一个接口。适配器使原本由于接口不兼容而不能一起工作的那些类可以一起工作。 9. **桥接模式**(Bridge...

    java设计模式与经验总结

    结构型模式如适配器(Adapter)、装饰器(Decorator)和代理(Proxy),关注于如何组合和构建对象。行为型模式如观察者(Observer)、策略(Strategy)和职责链(Chain of Responsibility),关注对象间的交互和行为...

    Java 的 23 种设计模式

    Java的`java.io.FilenameFilter`和`java.util.Predicate`是过滤器模式的例子。 11. **享元模式 (Flyweight)**:运用共享技术有效地支持大量细粒度的对象。Java的`java.lang.String`池就是享元模式的典型应用。 12....

    Head First设计模式 Java源代码

    Java的IO流系统就大量使用了装饰器模式。 8. **代理模式** (Proxy): 代理模式为其他对象提供一种代理以控制对这个对象的访问。Java中有两种代理:静态代理(手动实现接口)和动态代理(JDK或CGLIB实现)。 9. **...

    Java常用设计模式源码

    Java设计模式是软件开发中的重要概念,它是一种在特定情境下解决常见问题的经验总结,能够提升代码的可读性、可维护性和复用性。这个压缩包“Java常用设计模式源码”显然提供了多种常见的Java设计模式实现,适合初学...

Global site tag (gtag.js) - Google Analytics