什么是装饰器模式
装饰器模式通过组合替代继承的方式在不改变原始类的情况下添加增强功能,主要解决继承关系过于复杂的问题(Java IO 就属于这种复杂情况)。
刚上来,我们先知道装饰器是干啥的,解决啥问题就行,具体的是如何做的,我们边分析边说。
Java IO 庞大的类库
Java IO 的类库十分庞大,有 40 多个类,负责 IO 数据的读取和写入。我们可以从以下角度将其划分为四类,具体如下:
(抽象基类) | 字节流 | 字符流 |
---|---|---|
输入流 | InputStream | Reader |
输出流 | OutputStream | Writer |
针对不同的读取和写入场景,Java IO 又在四个父类基础上,扩展了很多子类。具体如下(只列举了一些常用的类):
Java IO 流的嵌套用法
还记得我们在 Java IO 基础篇中流的使用案例吗?若要使用缓存字节输入流,我们需要在 BufferedInputStream 的构造函数中传递一个 FileInputStream 对象来使用(这就是,使用 BufferedInputStream 增强 FileInputStream 的功能)。具体如下:
java
复制代码
try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream("test.txt"))) { byte[] b = new byte[128]; while (bis.read(b) != -1) { // ... } }
或许,你可能想为什么 Java IO 不设计一个继承 FileInputStream 并且支持缓存的 BufferedFileInputStream 类呢?
如果是这样的话,我们岂不是可以直接创建一个 BufferedFileInputStream 类对象,支持缓存并且可以打开文件读取数据,这样多省事简单啊。
java
复制代码
try (InputStream in = new BufferedFileInputStream("test.txt")) { byte[] b = new byte[128]; while (bis.read(b) != -1) { // ... } }
我们的这种思路就是基于继承的设计方案了。
基于继承的设计方案
如果说 InputStream
只有一个子类 FileInputStream
的话,那么我们在 InputStream
基础上,再设计一个孙子类 BufferedFileInputStream
,也算是可以,毕竟继承结构比较简单,能够接受。
然而,事实上,我们在上面的常用类图中也看到了,继承 InputStream
的子类非常多,那么我们就需要给每一个 InputStream
子类都派生一个支持缓存读取的子类,这数量太庞大了!
而且,支持缓存只是拓展功能之一,我们还要对其他功能进行增强,比如 DataInputStream
类,它支持按照所有 Java 基本数据类型来读取数据。
java
复制代码
try (DataInputStream dis = new DataInputStream(new FileInputStream("test.txt"))) { int da