目录:系统学习 Java IO---- 目录,概览

DataInputStream/DataOutputStream

允许应用程序以与机器无关方式从底层输入流中读取基本 Java 数据类型。

要想使用数据输出流和输入流,必须按指定的格式保存数据,才可以将数据输入流将数据读取进来,所以通常使用 DataInputStream 来读取 DataOutputStream 写入的数据。

DataInputStream 类能够从 InputStream 中读取 Java 基本类型(int,float,long等),而不仅仅是原始字节。 将InputStream包装在 DataInputStream 中,就可以从 DataInputStream 中读取 Java 基本类型。 这就是为什么它被称为 DataInputStream - 因为它读取数据(数字)而不仅仅是字节。

如果需要读取的数据包含大于一个字节的Java 基本类型,则 DataInputStream 非常方便。DataInputStream 希望接受有序多字节类型数据。

同时使用 DataInputStream 和 DataOutputStream

如前所述,DataInputStream 类通常与 DataOutputStream 一起使用,首先使用 DataOutputStream 写入数据,然后使用 DataInputStream 再次读取数据。 以下是示例Java代码:

public class DataStream {
public static void main(String[] args) throws IOException {
String file = "D:\\test\\output.txt";
DataOutputStream output = new DataOutputStream(new FileOutputStream(file));
output.write(1); // 默认是 byte
output.writeInt(123); // 指定写入 int
output.writeInt(321);
output.writeLong(789);
output.writeFloat(123.45f);
output.close(); // 一定要按照写入的顺序和类型读取,否则会出错;
DataInputStream input = new DataInputStream(new FileInputStream(file));
byte b = (byte) input.read();
int i1 = input.readInt();
int i2 = input.readInt();
Long l = input.readLong();
Float f = input.readFloat();
input.close(); System.out.println("i1 = " + i1);
System.out.println("i2 = " + i2);
System.out.println("b = " + b);
System.out.println("l = " + l);
System.out.println("f = " + f);
}
}

注意:一定要按照写入的顺序和类型读取,否则会出错;

其实 DataInputStream 类的实现中,读取方法中只有一个 read() 方法是真正干活,其他的 readXXX() 都是调用 read() 完成任务。看如下代码:

    public final byte readByte() throws IOException {
int ch = in.read();
if (ch < 0)
throw new EOFException();
return (byte)(ch);
} public final char readChar() throws IOException {
int ch1 = in.read();
int ch2 = in.read();
if ((ch1 | ch2) < 0)
throw new EOFException();
return (char)((ch1 << 8) + (ch2 << 0));
} public final int readInt() throws IOException {
int ch1 = in.read();
int ch2 = in.read();
int ch3 = in.read();
int ch4 = in.read();
if ((ch1 | ch2 | ch3 | ch4) < 0)
throw new EOFException();
return ((ch1 << 24) + (ch2 << 16) + (ch3 << 8) + (ch4 << 0));
}

可以看到,XXX 占 多少个字节,就会在其内部调用多少次 read() 方法。

ObjectInputStream/ObjectOutputStream

和 DataInputStream 包装成 Java 基本类型类似,ObjectInputStream 类能够从 InputStream 中读取Java对象,而不仅仅是原始字节。 当然,读取的字节必须表示有效的序列化 Java 对象。 通常,使用 ObjectInputStream 来读取 ObjectOutputStream 编写(序列化)的对象。

下面是一个例子:

public class ObjectStream {
public static void main(String[] args) throws Exception {
// Serializable 是一个标识接口,实现类只要承诺能被序列化就行了
class People implements Serializable {
String name;
int age;
}
File file = new File("D:\\test\\object.data"); ObjectOutputStream output = new ObjectOutputStream(new FileOutputStream(file));
People someOne = new People();
someOne.name = "Json";
someOne.age = 18;
output.writeObject(someOne);
output.close(); ObjectInputStream input = new ObjectInputStream(new FileInputStream(file));
People people = (People) input.readObject();
System.out.println("name = " + people.name + ", age = " + people.age);
input.close();
}
}
Close()

使用完数据流后记得关闭它。 关闭 DataInputStream 还将关闭 DataInputStream 正在读取的 InputStream 实例。可以使用 try-with-resources 方式自动关闭。ObjectInputStream 同理。

Serializable

如果一个类要进行序列化和反序列化,就必须实现 Serializable 标记接口,这样就可以使用 ObjectOutputStream 完成 Java 对象序列化(写入),使用 ObjectInputStream 完成反序列化(读取)。

Serializable 是一个标记接口意味着它不包含任何方法。 因此,实现 Serializable 的类不必实现任何特定方法,只是告诉 Java 该类对象支持序列化。

serialVersionUID

除了实现 Serializable 接口之外,用于序列化的类还应包含名为 serialVersionUID 的 private static final long 变量。

Java 的对象序列化 API 使用 serialVersionUID 变量来确定反序列化对象是否是使用相同版本的类进行序列化的,因为它现在正尝试将其反序列化。

想象一下,Person 对象被序列化为磁盘。 然后对 Person 类进行更改。 然后反序列化存储的 Person 对象。 这样,序列化的 Person 对象可能与 Person 类的新版本不对应。

要检测此类问题,实现 Serializable 的类应包含 serialVersionUID 字段。 如果对类进行了重大更改,则还应更改其 serialVersionUID 值。

许多 Java IDE 包含生成 serialVersionUID 的工具,可以使用工具生成的 UID 。

在今天的世界(2015年之后)中,许多 Java 项目使用与 Java 序列化机制不同的机制来序列化 Java 对象。 例如,Java 对象被序列化为 JSON,BSON 或其他更优化的二进制格式。 这具有以下优点:对象也可由非 Java 应用程序读取。 例如,在 Web 浏览器中运行的 JavaScript 可以本地序列化和反序列化 JSON 中的对象。

顺便说一下,这些其他对象序列化机制通常不需要 Java 类实现 Serializabl e。 他们通常使用 Java 反射来检查类,这里是 Java IO 教程,具体要看看 Java Json 的教程了。

系统学习 Java IO (十二)----数据流和对象流的更多相关文章

  1. 系统学习 Java IO (十六)----这么多类,应该用哪个?

    目录:系统学习 Java IO---- 目录,概览 Java IO目的和功能 Java IO 包含 InputStream,OutputStream,Reader 和 Writer 类的许多子类. 原 ...

  2. 系统学习 Java IO (十四)----字符读写缓存和回退 BufferedReader/BufferedWriter & PushbackReader

    目录:系统学习 Java IO---- 目录,概览 BufferedReader BufferedReader 类构造器接收一个 Reader 对象,为 Reader 实例提供缓冲. 缓冲可以加快 I ...

  3. 系统学习 Java IO (十五)----字符读写 Reader/Writer 其他子类

    目录:系统学习 Java IO---- 目录,概览 跟踪行号的缓冲字符输入流 LineNumberReader LineNumberReader 类是一个 BufferedReader ,用于跟踪读取 ...

  4. 系统学习 Java IO (十)----回退流 PushbackInputStream

    目录:系统学习 Java IO---- 目录,概览 PushbackInputStream 旨在从 InputStream 解析数据时使用. 有时您需要先读取几个字节以查看将要发生的事情,然后才能确定 ...

  5. 系统学习 Java IO ---- 目录,概览

    Java IO 类的系统教程,原创.主要参考自英文教程 Java IO Tutorial 和 Java Doc. http://tutorials.jenkov.com/java-io/index.h ...

  6. 系统学习 Java IO (一)----输入流和输出流 InputStream/OutputStream

    目录:系统学习 Java IO ---- 目录,概览 InputStream 是Java IO API中所有输入流的父类. 表示有序的字节流,换句话说,可以将 InputStream 中的数据作为有序 ...

  7. 系统学习 Java IO (八)----装饰流 FilterInputStream/FilterOutputStream

    目录:系统学习 Java IO---- 目录,概览 这两个流的作用是:"封装其它的输入流,并为它们提供额外的功能" 他们的直接子类有: BufferedInputStream 的作 ...

  8. 系统学习 Java IO (二)----IO 异常处理

    目录:系统学习 Java IO---- 目录,概览 我们使用流后,需要正确关闭 Streams 和 Readers / Writers . 这是通过调用 close() 方法完成的,看看下面这段代码: ...

  9. JAVA IO分析二:字节数组流、基本数据&对象类型的数据流、打印流

    上一节,我们分析了常见的节点流(FileInputStream/FileOutputStream  FileReader/FileWrite)和常见的处理流(BufferedInputStream/B ...

随机推荐

  1. Python Tricks(二十二)—— small tricks

    多次 import import numpy as np, matplotlib.pyplot as plt ndarray 的强制类型转换 v = v.astype(np.int) python 的 ...

  2. 非参贝叶斯(Bayesian Non-parameter)初步

    0. motivations 如何确定 GMM 模型的 k,既观察到的样本由多少个高斯分布生成.由此在数据属于高维空间中时,根本就无法 visualize,更加难以建立直观,从而很难确定 k,高斯分布 ...

  3. [WPF] PerformClick ?

    原文:[WPF] PerformClick ? [WPF] PerformClick ?  周银辉 WPF没有提供这个方法,还真是让人觉得有些讨厌啊.而关于这个嘛,Google中搜一下,一大堆,但一般 ...

  4. Android 4.0屏蔽式多点触摸

    比方这张图.我想不接或者接单,二者仅仅能点一个,不能同一时候点击,否则会造成混乱.我们仅仅要在嵌套他们俩的布局中增加这么一段话: android:splitMotionEvents="fal ...

  5. RPC的发展历史(本质就是双方定义好协议,传递参数后远程调用)

    服务器通讯原理就是一台socket服务器A,另一台socket客户端B,现在如果要通讯的话直接以流方式写入或读出. 这样能实现通讯,但有个问题.如何知道更多信息?比如需要发送流大小,编码,Ip等. 这 ...

  6. Microsoft.AspNet.SignalR实现弹幕(即时通讯)

    引用 Microsoft.AspNet.SignalR 服务器 自定义Connection public class BarrageConnection : PersistentConnection ...

  7. NPM镜像设置方法!

    使用npm安装一些包失败了的看过来(npm国内镜像介绍) 发布于 2012-4-26 04:19 最后一次编辑是 2013-12-11 23:21 这个也是网上搜的,亲自试过,非常好用! 镜像使用方法 ...

  8. WPF UpdateSourceTrigger的使用

    <Window x:Class="XamlTest.Window8"        xmlns="http://schemas.microsoft.com/winf ...

  9. EF 两种删除方式的比较

    UserInfo user = from u in context.UserInfo where u.Id=343 select u; context.UserInfo.Remove(user); 用 ...

  10. WPF 自定义范围分组

    <Window x:Class="ViewExam.MainWindow"        xmlns="http://schemas.microsoft.com/w ...