Java中的输入输出流(I_O)详解
发布时间: 2023-12-19 21:54:40 阅读量: 31 订阅数: 36
# 1. 简介
## 1.1 什么是输入输出流(I_O)
输入输出流(I/O)是用于读取数据和写入数据的抽象概念。在计算机中,所有设备(如磁盘驱动器、键盘、显示器等)都被视为一个流,数据从一个流传输到另一个流。输入流用于从设备中读取数据,输出流用于向设备中写入数据。
## 1.2 输入输出流的作用
输入输出流的主要作用是从源读取数据并将数据传输到目标。数据源可以是文件、内存、网络连接或其他设备,而目标可以是文件、内存、网络连接或打印机等。
## 1.3 输入输出流的类型
输入输出流可以分为字节流和字符流。字节流以字节为单位读取和写入数据,适用于处理二进制数据,字符流以字符为单位读取和写入数据,适用于处理文本数据。
在接下来的章节中,我们将深入探讨输入流、输出流、字符流以及缓冲流的概念、应用和示例。
# 2. 输入流(InputStream)
输入流(InputStream)是Java I/O中用于从数据源中读取数据的抽象类。它是所有输入流的基类,提供了一些基本的方法用于读取字节数据。
### 2.1 InputStream的基本概念
InputStream提供了以下几种基本的方法:
- `int read()`: 从输入流中读取一个字节的数据,并返回读取的字节(0-255),如果已达到输入流的末尾,则返回-1。
- `int read(byte[] b)`: 从输入流中读取数据到字节数组b中,并返回实际读取的字节数。如果已达到输入流的末尾,则返回-1。
- `int read(byte[] b, int off, int len)`: 从输入流中读取数据到字节数组b中的指定位置,并返回实际读取的字节数。off表示偏移量,len表示读取的字节数。如果已达到输入流的末尾,则返回-1。
- `long skip(long n)`: 跳过n个字节的数据。
- `int available()`: 返回输入流中可读取的字节数。
- `void close()`: 关闭输入流。
### 2.2 InputStream的常用子类
Java提供了几个常用的InputStream的子类,用于读取不同的数据源,包括文件、网络连接、内存等。
- FileInputStream:用于读取文件的输入流。
- ByteArrayInputStream:用于读取内存中的数据的输入流。
- SocketInputStream:用于读取网络套接字的输入流。
- ...
### 2.3 InputStream的使用示例
下面是一个使用FileInputStream读取文件的示例代码:
```java
import java.io.FileInputStream;
import java.io.IOException;
public class InputStreamExample {
public static void main(String[] args) {
FileInputStream fis = null;
try {
fis = new FileInputStream("input.txt");
int data;
while ((data = fis.read()) != -1) {
System.out.print((char) data);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
```
上述代码中,我们使用FileInputStream来读取名为`input.txt`的文件。通过`fis.read()`方法读取文件中的字节数据,并将其转换为字符输出。最后,再通过`fis.close()`方法关闭输入流。
这是一个简单的输入流示例,通过这种方式我们可以读取文件中的数据并进行处理。在实际开发中,我们可以根据需要选择合适的子类来读取不同的数据源。
# 3. 输出流(OutputStream)
#### 3.1 OutputStream的基本概念
在Java中,OutputStream是抽象类,用于表示字节输出流。它是所有字节输出流的父类,提供了写入字节的基本方法。
#### 3.2 OutputStream的常用子类
一些常用的子类包括:
- `FileOutputStream`:用于将数据写入文件。
- `ByteArrayOutputStream`:在内存中创建一个字节数组缓冲区,所有数据将写入到这个缓冲区中。
- `DataOutputStream`:用于将Java基本数据类型写入输出流。
#### 3.3 OutputStream的使用示例
下面是一个简单的示例,演示了如何使用`FileOutputStream`将数据写入文件:
```java
import java.io.*;
public class OutputStreamExample {
public static void main(String[] args) {
try {
String data = "Hello, Output Stream!";
FileOutputStream fileOutput = new FileOutputStream("output.txt");
byte[] byteData = data.getBytes();
fileOutput.write(byteData);
fileOutput.close();
System.out.println("Data has been written to the file.");
} catch (IOException e) {
e.printStackTrace();
}
}
}
```
在这个示例中,我们首先创建了一个`FileOutputStream`对象,并指定了要写入数据的文件名。然后我们将字符串转换为字节数组,最后使用`write`方法将数据写入文件。最后别忘记关闭输出流。
这就是`OutputStream`的简单使用示例。
希望这个内容符合你的需求,如果还需要其他方面的详细内容,请随时告诉我。
# 4. 字符流(Reader和Writer)
### 4.1 Reader和Writer的介绍
在之前的章节中,我们已经介绍了输入流和输出流的概念和用法。而在处理文本数据时,字符流是更加方便和高效的选择。字符流中的Reader和Writer类是专门用于处理字符数据的输入和输出流。
Reader和Writer是Java中的抽象类,它们分别位于java.io包下。
- Reader:用于读取字符数据流。
- Writer:用于写入字符数据流。
与字节流不同的是,字符流的操作单位是字符,而不是字节。字符流能够方便地处理Unicode字符和文本数据。
### 4.2 Reader和Writer的常用子类
Reader和Writer类是抽象类,通常使用它们的具体子类来进行字符数据的读写操作。一些常用的Reader和Writer子类有:
- BufferedReader:提供了缓冲功能,可以一次读取多个字符。
- InputStreamReader:将字节流转换为字符流的桥梁。
- FileWriter:用于将字符数据写入文件。
- CharArrayReader:从字符数组中读取数据。
- StringWriter:将字符数据写入字符串缓冲区。
这些子类都提供了不同的功能和方便的方法,可以根据实际需求选择合适的子类进行操作。
### 4.3 Reader和Writer的使用示例
接下来,我们将通过一个示例来演示如何使用Reader和Writer进行字符数据的读写操作。
```java
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
public class CharacterStreamExample {
public static void main(String[] args) {
try {
// 从文件读取字符数据
FileReader reader = new FileReader("input.txt");
int data;
while ((data = reader.read()) != -1) {
System.out.print((char) data);
}
reader.close();
// 将字符数据写入文件
FileWriter writer = new FileWriter("output.txt");
writer.write("Hello, World!");
writer.close();
System.out.println("\n数据写入成功!");
} catch (IOException e) {
e.printStackTrace();
}
}
}
```
在上面的示例中,我们首先使用FileReader读取文件中的字符数据,并使用while循环逐个字符进行输出。然后,我们使用FileWriter将字符串"Hello, World!"写入到output.txt文件中。
请注意,在使用字符流进行读写操作时,需要注意异常处理和流的关闭操作以确保资源的正确释放。
以上就是使用字符流(Reader和Writer)进行读写操作的示例,通过字符流的使用,我们可以方便地处理文本数据,对于需要进行字符操作的场景非常有用。
总结:字符流(Reader和Writer)是Java中处理字符数据的输入和输出流,通过使用字符流,我们可以方便地进行文本数据的读取和写入操作。
# 5. 缓冲流(BufferedInputStream和BufferedOutputStream)
缓冲流可以提高输入输出流的性能,通过在内存中设立缓冲区,减少了直接和目标设备的交互次数,进而提高了读写的效率。
#### 5.1 缓冲流的作用
缓冲流主要用于提高输入输出流的读写效率。它通过在内存中创建缓冲区,将数据先写入缓冲区,再一次性写入输出流,或者先将数据从输入流读入缓冲区,再从缓冲区一次性读取数据,避免频繁的磁盘IO操作,提高读写速度。
#### 5.2 BufferedInputStream和BufferedOutputStream的介绍
- **BufferedInputStream**:BufferedInputStream继承自FilterInputStream类,提供了缓冲输入流的功能,它可以从输入流中读取字节并缓冲每次读取的数据。
- **BufferedOutputStream**:BufferedOutputStream继承自FilterOutputStream类,提供了缓冲输出流的功能,它可以将数据写入缓冲区,并一次性将缓冲区的数据写入输出流。
#### 5.3 缓冲流的使用示例
下面是Java中使用缓冲流进行文件复制的示例代码:
```java
import java.io.*;
public class FileCopyWithBufferedStream {
public static void main(String[] args) {
String sourceFile = "source.txt";
String destinationFile = "destination.txt";
try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream(sourceFile));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(destinationFile))) {
byte[] buffer = new byte[1024];
int bytesRead;
while ((bytesRead = bis.read(buffer)) != -1) {
bos.write(buffer, 0, bytesRead);
}
System.out.println("File copy successful");
} catch (IOException e) {
e.printStackTrace();
}
}
}
```
在这个示例中,我们利用BufferedInputStream和BufferedOutputStream来完成文件的复制操作。
以上是缓冲流的介绍和使用示例,它可以帮助提高输入输出流的读写性能。
# 6. 序列化与反序列化
在程序中,我们经常需要将对象转换为字节流,以便在网络上传输或者保存到文件中。而序列化与反序列化就是实现这样的功能。在Java中,序列化和反序列化是通过实现Serializable接口来实现的。
### 6.1 什么是序列化与反序列化
#### 序列化
序列化是指将对象转换为字节流的过程。在序列化过程中,对象的状态信息也会被保存下来,包括对象的属性值以及对象的类型信息。序列化后的字节流可以保存到文件中或者通过网络传输。
#### 反序列化
反序列化是指将字节流转换为对象的过程。在反序列化过程中,字节流会被还原为之前序列化时的对象,即恢复对象的状态信息。
### 6.2 Java中序列化与反序列化的实现方式
Java提供了两种实现序列化和反序列化的方式,分别是使用Serializable接口和Externalizable接口。
#### Serializable接口
Serializable接口是Java内置的接口,只需要在需要被序列化的类中实现Serializable即可,如下所示:
```java
import java.io.Serializable;
public class Person implements Serializable {
private static final long serialVersionUID = 1L;
private String name;
private int age;
// 省略构造方法和其他属性的getter和setter方法
}
```
#### Externalizable接口
Externalizable接口是Java内置的接口,需要在需要被序列化的类中实现Externalizable接口,并重写readExternal()和writeExternal()方法,如下所示:
```java
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
public class Person implements Externalizable {
private String name;
private int age;
// 省略构造方法和其他属性的getter和setter方法
@Override
public void writeExternal(ObjectOutput out) throws IOException {
// 实现对象的序列化
out.writeObject(this.name);
out.writeInt(this.age);
}
@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
// 实现对象的反序列化
this.name = (String) in.readObject();
this.age = in.readInt();
}
}
```
### 6.3 序列化与反序列化的注意事项
在进行序列化和反序列化时,需要注意以下几点:
- 序列化的类必须实现Serializable接口或者Externalizable接口。
- 被序列化的类的属性也必须是可序列化的,否则需要标记为transient关键字。
- 序列化和反序列化的类的class文件必须是一致的,否则会抛出InvalidClassException异常。
- 对象的实例变量是无法被序列化的,只有对象的状态才能被序列化。
- 在进行网络传输时,需要保证发送方和接收方都使用相同的序列化方式,否则会发生错误。
综上所述,序列化和反序列化是实现对象与字节流之间转换的重要方式。通过序列化可以将对象转换为字节流,从而实现对象的保存和网络传输。而通过反序列化可以将字节流还原为对象,从而恢复对象的状态信息。在实际应用中,我们必须注意序列化和反序列化的方式以及使用的注意事项,以确保数据的正确传输和恢复。
0
0