java.io 这个包里有一个类,比较特别,这就是BufferedReader。我们从JDK的源码里,找到它的实现:
public class BufferedReader extends Reader {
public BufferedReader(Reader in, int sz) {
super(in);
if (sz <= 0)
throw new IllegalArgumentException("Buffer size <= 0");
this.in = in;
cb = new char[sz];
nextChar = nChars = 0;
}
}
可以看到,BufferedReader本身就是一个Reader,因为它继承自Reader,同时,还有一个名为 in 的成员变量,也是Reader类型的,然后还开辟了一个数组。实际上,这个数组是为了做缓存的,我们可以一次从 in 这个成员对象中读取多个字符存入 cb 中。当真正调用BufferedReader 的 read 方法的时候,就直接从 cb 中读取了,提高了读取的性能。也就是说BufferedReader所提供的 read 方法经过了 cb 这个缓存的加速,其性能会高于直接从 in 这个对象去读取。这是一种增强普通的 Reader 对象的技术。
注意一点,BufferedReader在对一个Reader的对象做增强的时候,只要求了对象是Reader类型的,并没有要求对象是具体哪个类型。这个in对象可以是SocketReader,也可以是FileReader。
这种本身是一种类型,保持类型接口不变,又能对该类型的其他子类进行加强的能力,就叫装饰者。就好像,我们在一个对象上面增加一种装饰,使得它更好看了,或者功能更强大了。看完了代码再看类图就比较清楚了:
可见,decorator模式,是一种可以为对象增加额外能力的设计模式。
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
我们的完整版。从此以后,就可以直接使用 br 了。即可以使用 Reader 接口规定的read方法,也可以使用 BufferedReader 自己增强的 readLine方法,一次读入一行字符串。
看,小小的一行代码,包含了适配器和装饰器。Java 的 IO的全部知识都在这里了。从设计模式的角度去学习输入输出,就像庖丁手里的那把刀,真是 “恢恢乎其于游刃必有余地矣”。