# IO

数据流是一组有序，有起点和终点的字节的数据序列。包括输入流和输出流。

当程序需要读取数据的时候，就会建立一个通向数据源的连接，这个数据源可以是文件，内存，或是网络连接。类似的，当程序需要写入数据的时候，就会建立一个通向目的地的连接。

`Java.io`包中最重要的就是5个类，指的是`File、OutputStream、InputStream、Writer、Reader`。

![Java IO 体系](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LZiAuukbEZZ1VM18dik%2F-LZiC2mrlRO8hXHddpOe%2Fimage.png?alt=media\&token=6a8e565a-4c01-4165-b97f-549721236903)

## Stream

![按操作方式分类](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LZiDA98OzCeEToFZXes%2F-LZiFiXaz4Mk8KxpjJ22%2Fimage.png?alt=media\&token=71f65f7c-5f39-4760-861e-b890bc0154d8)

![按操作对象分类](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LZiDA98OzCeEToFZXes%2F-LZiI8NLTu3dkHABALyw%2Fimage.png?alt=media\&token=23f22809-8b46-4eaf-8edd-4c116e239463)

### 字节流

数据流中最小的数据单元是**字节**。

![字节流输入输出对应关系](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LZiDA98OzCeEToFZXes%2F-LZiG24mv2KIzFu57eaq%2Fimage.png?alt=media\&token=4092c359-7822-4c3c-affa-e9755d137e3f)

**`InputStream`：**&#x5B57;节流，二进制格式操作，抽象类，基于字节的输入操作，是所有输入流的父类。定义了所有输入流都具有的共同特征。

**`OutputStream`：**&#x5B57;节流，二进制格式操作，抽象类。基于字节的输出操作。是所有输出流的父类。定义了所有输出流都具有的共同特征。

### 对象序列化

ObjectOutputStream 和 ObjectInputStream 是实现对象序列化与反序列的类，只针对实现了 Serializable 接口的类。不会对静态变量和 transient 变量序列化。类有一个 serialVersionUID 变量，表示类的版本。

![](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LgpPhjpJSTmBz1D-n2G%2F-LgpQR1WTAimxObhCENc%2Fimage.png?alt=media\&token=fb165ae1-44e0-495e-bbbb-db1e13cc5bc3)

Java 内置的序列化方式有很多缺点：

* 无法跨语言。
* 易被攻击。
* 序列化后的字节很大。
* 性能差。

所以有很多第三方序列化实现，比如：FastJson、Kryo、Protobuf、Hessian 等。

### 字符流

数据流中最小的数据单元是**字符**， Java 中的字符是 Unicode 编码，一个字符占用两个字节。

![字符流输入与输出对应关系](https://3232244687-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-LYZow-MmROshIrkwdtE%2F-LZiDA98OzCeEToFZXes%2F-LZiHhN3ciisl3u8RvUi%2Fimage.png?alt=media\&token=85ecdc5b-718c-4607-a705-368eca23bb67)

**`Reader`：**&#x5B57;符流，文本格式操作，抽象类，基于字符的输入操作。

**`Writer`：**&#x5B57;符流，文本格式操作，抽象类，基于字符的输出操作。

### 选择 IO 流

**输入还是输出：**

* 输入：输入流 InputStream Reader
* 输出：输出流 OutputStream Writer

**操作的数据对象是否是纯文本：**

* 是：字符流 Reader，Writer
* 否：字节流 InputStream，OutputStream

**具体的设备**：

* **文件**：
  * 读：FileInputStream,, FileReader,
  * 写：FileOutputStream，FileWriter
* **数组**：
  * byte\[ ]：ByteArrayInputStream, ByteArrayOutputStream
  * char\[ ]：CharArrayReader, CharArrayWriter
* **String**：
  * StringBufferInputStream (已过时，因为其只能用于 String 的每个字符都是8位的字符串), StringReader, StringWriter
* **Socket** 流：
  * 键盘：用 System.in（是一个InputStream对象）读取，用 System.out（是一个 OutputStream 对象）打印

**是否需要转换流：**

* 是，就使用转换流，从 Stream 转化为 Reader、Writer：InputStreamReader，OutputStreamWriter。

**是否需要缓冲提高效率：**

* 是就加上 Buffered：BufferedInputStream, BufferedOuputStream, BufferedReader, BufferedWriter。

## 文件类

**`File`：**&#x6587;件特征与管理，用于文件或者目录的描述信息，例如生成新目录，修改文件名，删除文件，判断文件所在路径等。

**`RandomAccessFile`：**&#x968F;机文件操作，它的功能丰富，可以从文件的任意位置进行存取（输入输出）操作。
