对象流
原文: https://docs.oracle.com/javase/tutorial/essential/io/objectstreams.html
正如数据流支持原始数据类型的 I / O 一样,对象流也支持对象的 I / O.大多数(但不是全部)标准类支持其对象的序列化。那些确实实现了标记接口 Serializable
。
对象流类是 ObjectInputStream
和 ObjectOutputStream
。这些类实现 ObjectInput
和 ObjectOutput
,它们是DataInput
和DataOutput
的子接口。这意味着数据流中涵盖的所有原始数据 I / O 方法也在对象流中实现。因此,对象流可以包含原始值和对象值的混合。 ObjectStreams
的例子说明了这一点。 ObjectStreams
创建与DataStreams
相同的应用程序,但有一些更改。首先,价格现在是 BigDecimal
对象,以更好地表示小数值。其次,将 Calendar
对象写入数据文件,指示发票日期。
如果readObject()
未返回预期的对象类型,则尝试将其转换为正确的类型可能会抛出 ClassNotFoundException
。在这个简单的例子中,这不可能发生,因此我们不会尝试捕获异常。相反,我们通过将ClassNotFoundException
添加到main
方法的throws
子句来通知编译器我们已经知道了这个问题。
复杂对象的输出和输入
writeObject
和readObject
方法易于使用,但它们包含一些非常复杂的对象管理逻辑。这对像 Calendar 这样的类来说并不重要,它只封装了原始值。但是许多对象包含对其他对象的引用。如果readObject
要从流重构对象,则它必须能够重建原始对象所引用的所有对象。这些附加对象可能有自己的引用,依此类推。在这种情况下,writeObject
遍历整个对象引用 Web,并将该 Web 中的所有对象写入流。因此,单个writeObject
调用可能会导致大量对象被写入流。
这在下图中进行了演示,其中调用writeObject
来编写名为 a 的单个对象。该对象包含对象 b 和 c 的参考,而 b 包含对 d 和 e 的参考。调用writeobject(a)
不仅会写一个,而且还需要重建和所需的所有对象,因此也会写入此 Web 中的其他四个对象。当[COG2]回读和时,也会回读其他四个对象,并保留所有原始对象引用。
多个引用对象的 I / O.
您可能想知道如果同一个流上的两个对象都包含对单个对象的引用会发生什么。当他们回读时,他们都会引用一个对象吗?答案是肯定的。流只能包含一个对象的副本,尽管它可以包含任意数量的对象。因此,如果您将对象显式写入流两次,那么您实际上只编写了两次引用。例如,如果以下代码将对象ob
两次写入流:
Object ob = new Object();
out.writeObject(ob);
out.writeObject(ob);
每个writeObject
必须与readObject
匹配,因此读取流的代码将如下所示:
Object ob1 = in.readObject();
Object ob2 = in.readObject();
这导致两个变量ob1
和ob2
,它们是对单个对象的引用。
但是,如果将单个对象写入两个不同的流,则它实际上是重复的 - 读取两个流的单个程序将看到两个不同的对象。