文章目录
- 类层次结构
- 1. `java.io` 包
- 整体示意图
- 核心类
- 字节流(Byte Stream)
- 字符流(Character Stream)
- 其他辅助类
- 2. `java.nio` 包
- 示意图
- 核心类
- 文件操作
- 总结
- 字符流读写
- 1. 字符流的核心类
- 1.1 `Reader` 和 `Writer`
- 2. 常用的字符流类
- 2.1 字符输入流
- 2.2 字符输出流
- 3. 示例代码
- 3.1 读取文本文件
- 3.2 写入文本文件
- 4. 使用 `InputStreamReader` 和 `OutputStreamWriter`
- 4.1 从字节流读取字符
- 4.2 向字节流写入字符
- 5. 使用 `PrintWriter`
- 总结
- 字节流读写
- 1. 字节流的核心类
- 1.1 输入流
- 1.2 输出流
- 2. 示例代码
- 2.1 读取二进制文件
- 2.2 写入二进制文件
- 2.3 使用 `DataInputStream` 和 `DataOutputStream`
- 3. 使用 `InputStreamReader` 和 `OutputStreamWriter`
- 总结
- nio读写
- 1. NIO 的核心组件
- 1.1 缓冲区(Buffer)
- 1.2 通道(Channel)
- 1.3 选择器(Selector)
- 2. 示例代码
- 2.1 读取文本文件
- 2.2 写入文本文件
- 2.3 使用 `FileChannel` 读写文件
- 2.4 使用 `Selector` 进行网络编程
- 总结
类层次结构
Java 的 I/O 类库提供了丰富的 API 来处理输入/输出操作,包括文件读写、网络通信等。Java 的 I/O 类库主要分为两个部分:java.io
包和 java.nio
包。下面详细介绍这两个包中的核心类及其结构。
1. java.io
包
java.io
包是 Java 最初提供的 I/O 操作基础库,它主要基于流(Stream)的概念。java.io
包中的类大多采用阻塞式的 I/O 模型。
整体示意图
核心类
- InputStream/OutputStream:所有的字节流类都是从这两个抽象类派生出来的。
InputStream
:表示字节输入流。OutputStream
:表示字节输出流。
- Reader/Writer:所有的字符流类都是从这两个抽象类派生出来的。
Reader
:表示字符输入流。Writer
:表示字符输出流。
字节流(Byte Stream)
- FileInputStream/FileOutputStream:用于读写文件。
- ByteArrayInputStream/ByteArrayOutputStream:用于在内存中读写字节数组。
- PipedInputStream/PipedOutputStream:用于线程间的通信。
- FilterInputStream/FilterOutputStream:过滤流,用于包装其他流并提供附加功能。
字符流(Character Stream)
- FileReader/FileWriter:用于读写文件,以字符为单位。
- StringReader/StringWriter:用于在内存中读写字符串。
- BufferedReader/BufferedWriter:带缓冲区的字符流,提高读写效率。
- InputStreamReader/OutputStreamWriter:用于将字节流转换为字符流。
- PrintWriter:用于格式化输出。
其他辅助类
- DataInputStream/DataOutputStream:用于读写基本数据类型。
- ObjectInputStream/ObjectOutputStream:用于序列化和反序列化对象。
- SequenceInputStream:合并多个输入流。
2. java.nio
包
java.nio
包提供了基于通道(Channel)和缓冲区(Buffer)的新 I/O 操作,主要用于提高 I/O 操作的性能。java.nio
包中的类通常是非阻塞式的,更适合于处理大量并发连接的场景。
示意图
核心类
-
Buffer:缓冲区,用于存储数据。
ByteBuffer
:存储字节数据。CharBuffer
:存储字符数据。IntBuffer
:存储整型数据。FloatBuffer
:存储浮点型数据。- 等等。
-
Channel:通道,用于连接 Buffer 和实际的 I/O 设备。
FileChannel
:用于文件 I/O。DatagramChannel
:用于 UDP 通信。SocketChannel
:用于 TCP 通信。ServerSocketChannel
:用于监听 TCP 连接。
-
Selector:选择器,用于多路复用 I/O 操作,管理多个 Channel。
文件操作
Files
:提供静态方法来简化文件操作。Paths
:提供静态方法来创建和操作路径。Path
:表示文件系统的路径。
总结
java.io
包提供了一系列基于流的 I/O 操作,适用于简单的文件读写和网络通信。而 java.nio
包则提供了更高性能的 I/O 操作,适用于需要处理大量并发连接的应用场景。在实际开发中,可以根据具体的性能需求和技术背景选择合适的 I/O 模型。
字符流读写
Java 中的字符流(Character Stream)主要用于处理文本数据,因为它们以字符而不是字节的形式处理数据。字符流通常使用 java.io
包中的 Reader
和 Writer
类及其子类来实现。字符流非常适合用于处理文本文件,因为它们可以方便地处理 Unicode 字符。
1. 字符流的核心类
1.1 Reader
和 Writer
Reader
:这是所有字符输入流的超类。Writer
:这是所有字符输出流的超类。
2. 常用的字符流类
2.1 字符输入流
FileReader
:用于从文件中读取字符。StringReader
:用于从字符串中读取字符。BufferedReader
:带缓冲的字符输入流,提高了读取效率。InputStreamReader
:用于将字节流转换成字符流。
2.2 字符输出流
FileWriter
:用于向文件中写入字符。StringWriter
:用于向字符串中写入字符。BufferedWriter
:带缓冲的字符输出流,提高了写入效率。OutputStreamWriter
:用于将字符流转换成字节流。PrintWriter
:用于格式化输出,常用于标准输出。
3. 示例代码
3.1 读取文本文件
使用 FileReader
和 BufferedReader
读取文本文件:
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;public class ReadTextFile {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try (BufferedReader reader = new BufferedReader(new FileReader(filePath))) {String line;while ((line = reader.readLine()) != null) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();}}
}
3.2 写入文本文件
使用 FileWriter
和 BufferedWriter
写入文本文件:
import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;public class WriteTextFile {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try (BufferedWriter writer = new BufferedWriter(new FileWriter(filePath))) {writer.write("Hello, World!");writer.newLine(); // 添加新行writer.write("This is a test file.");} catch (IOException e) {e.printStackTrace();}}
}
4. 使用 InputStreamReader
和 OutputStreamWriter
有时候,我们需要将字节流转换为字符流,这时可以使用 InputStreamReader
和 OutputStreamWriter
。
4.1 从字节流读取字符
使用 InputStreamReader
从字节流中读取字符:
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.FileInputStream;
import java.io.IOException;public class ReadFromByteStream {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(filePath)))) {String line;while ((line = reader.readLine()) != null) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();}}
}
4.2 向字节流写入字符
使用 OutputStreamWriter
向字节流中写入字符:
import java.io.BufferedWriter;
import java.io.OutputStreamWriter;
import java.io.FileOutputStream;
import java.io.IOException;public class WriteToByteStream {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try (BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(filePath)))) {writer.write("Hello, World!");writer.newLine(); // 添加新行writer.write("This is a test file.");} catch (IOException e) {e.printStackTrace();}}
}
5. 使用 PrintWriter
PrintWriter
类提供了一些方便的方法来格式化输出,通常用于标准输出或文件输出。
import java.io.PrintWriter;
import java.io.FileWriter;
import java.io.IOException;public class UsePrintWriter {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try (PrintWriter writer = new PrintWriter(new FileWriter(filePath))) {writer.println("Hello, World!");writer.println("This is a test file.");} catch (IOException e) {e.printStackTrace();}}
}
总结
字符流非常适合处理文本数据,特别是在需要处理 Unicode 字符时。通过使用 Reader
和 Writer
及其子类,可以方便地实现文本文件的读写操作。此外,通过使用 InputStreamReader
和 OutputStreamWriter
,还可以将字节流转换为字符流,从而更好地处理文本数据。
字节流读写
Java 中的字节流(Byte Stream)主要用于处理二进制数据,如图像、音频、视频等非文本数据。字节流是以字节为单位进行读写的,通常使用 java.io
包中的 InputStream
和 OutputStream
类及其子类来实现。
1. 字节流的核心类
1.1 输入流
InputStream
:这是所有字节输入流的超类。FileInputStream
:用于从文件中读取字节。ByteArrayInputStream
:用于从字节数组中读取字节。PipedInputStream
:用于线程间通信。FilterInputStream
:过滤流,用于包装其他流并提供附加功能。BufferedInputStream
:带缓冲区的字节输入流,提高了读取效率。DataInputStream
:用于读取基本数据类型。
1.2 输出流
OutputStream
:这是所有字节输出流的超类。FileOutputStream
:用于向文件中写入字节。ByteArrayOutputStream
:用于向字节数组中写入字节。PipedOutputStream
:用于线程间通信。FilterOutputStream
:过滤流,用于包装其他流并提供附加功能。BufferedOutputStream
:带缓冲区的字节输出流,提高了写入效率。DataOutputStream
:用于写入基本数据类型。
2. 示例代码
2.1 读取二进制文件
使用 FileInputStream
和 BufferedInputStream
读取二进制文件:
import java.io.FileInputStream;
import java.io.BufferedInputStream;
import java.io.IOException;public class ReadBinaryFile {public static void main(String[] args) {String filePath = "/path/to/your/file.bin";try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream(filePath))) {int data;while ((data = bis.read()) != -1) {System.out.printf("%02X ", data); // 以十六进制格式打印每个字节}} catch (IOException e) {e.printStackTrace();}}
}
2.2 写入二进制文件
使用 FileOutputStream
和 BufferedOutputStream
写入二进制文件:
import java.io.FileOutputStream;
import java.io.BufferedOutputStream;
import java.io.IOException;public class WriteBinaryFile {public static void main(String[] args) {String filePath = "/path/to/your/file.bin";byte[] bytes = {0x00, 0x01, 0x02, 0x03, 0x04};try (BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(filePath))) {bos.write(bytes);} catch (IOException e) {e.printStackTrace();}}
}
2.3 使用 DataInputStream
和 DataOutputStream
使用 DataInputStream
和 DataOutputStream
来读写基本数据类型:
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;public class DataIOExample {public static void main(String[] args) {String filePath = "/path/to/your/datafile.bin";// 写入数据try (DataOutputStream dos = new DataOutputStream(new FileOutputStream(filePath))) {dos.writeInt(12345); // 写入整数dos.writeDouble(3.14); // 写入双精度浮点数dos.writeUTF("Hello, World!"); // 写入字符串} catch (IOException e) {e.printStackTrace();}// 读取数据try (DataInputStream dis = new DataInputStream(new FileInputStream(filePath))) {int intValue = dis.readInt(); // 读取整数double doubleValue = dis.readDouble(); // 读取双精度浮点数String stringValue = dis.readUTF(); // 读取字符串System.out.println("Read values:");System.out.println("Integer: " + intValue);System.out.println("Double: " + doubleValue);System.out.println("String: " + stringValue);} catch (IOException e) {e.printStackTrace();}}
}
3. 使用 InputStreamReader
和 OutputStreamWriter
有时候,需要将字节流转换为字符流,可以使用 InputStreamReader
和 OutputStreamWriter
:
import java.io.InputStream;
import java.io.OutputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;public class ByteToCharConversion {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";// 读取字符try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(filePath)))) {String line;while ((line = reader.readLine()) != null) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();}// 写入字符try (BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(filePath)))) {writer.write("Hello, World!");writer.newLine(); // 添加新行writer.write("This is a test file.");} catch (IOException e) {e.printStackTrace();}}
}
总结
字节流非常适合处理二进制数据,通过使用 InputStream
和 OutputStream
及其子类,可以方便地实现二进制文件的读写操作。此外,通过使用 DataInputStream
和 DataOutputStream
,还可以方便地读写基本数据类型。如果需要将字节流转换为字符流,可以使用 InputStreamReader
和 OutputStreamWriter
。
nio读写
Java NIO(New Input/Output)是 Java 平台上的新一代输入/输出技术,它提供了更高效的 I/O 操作,特别是针对大数据量和高并发的应用场景。NIO 引入了缓冲区(Buffer)、通道(Channel)和选择器(Selector)等概念,使得 I/O 操作更加灵活和高效。
1. NIO 的核心组件
1.1 缓冲区(Buffer)
缓冲区用于存储数据,是 NIO 中进行数据读写的基础。常用的缓冲区包括:
ByteBuffer
:用于存储字节数据。CharBuffer
:用于存储字符数据。IntBuffer
:用于存储整型数据。FloatBuffer
:用于存储浮点型数据。DoubleBuffer
:用于存储双精度浮点型数据。ShortBuffer
:用于存储短整型数据。LongBuffer
:用于存储长整型数据。
1.2 通道(Channel)
通道用于连接缓冲区和实际的 I/O 设备(如磁盘、网络等)。常用的通道包括:
FileChannel
:用于文件 I/O。DatagramChannel
:用于 UDP 通信。SocketChannel
:用于 TCP 通信。ServerSocketChannel
:用于监听 TCP 连接。
1.3 选择器(Selector)
选择器用于多路复用 I/O 操作,可以同时监听多个通道上的事件(如可读、可写等)。
2. 示例代码
2.1 读取文本文件
使用 NIO 读取文本文件:
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.nio.charset.StandardCharsets;public class ReadTextFileWithNIO {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";try {// 使用 Files.readAllLines 读取整个文件List<String> lines = Files.readAllLines(Paths.get(filePath), StandardCharsets.UTF_8);for (String line : lines) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();}}
}
2.2 写入文本文件
使用 NIO 写入文本文件:
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.nio.charset.StandardCharsets;public class WriteTextFileWithNIO {public static void main(String[] args) {String filePath = "/path/to/your/file.txt";String content = "Hello, World!\nThis is a test file.";try {// 使用 Files.write 写入整个文件Files.write(Paths.get(filePath), content.getBytes(StandardCharsets.UTF_8));} catch (IOException e) {e.printStackTrace();}}
}
2.3 使用 FileChannel
读写文件
使用 FileChannel
和 Buffer
读写文件:
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;public class FileChannelExample {public static void main(String[] args) {String filePath = "/path/to/your/file.bin";// 写入数据try (FileChannel channel = FileChannel.open(Paths.get(filePath), StandardOpenOption.WRITE, StandardOpenOption.CREATE)) {ByteBuffer buffer = ByteBuffer.allocate(1024); // 创建缓冲区buffer.put("Hello, World!".getBytes()); // 将数据放入缓冲区buffer.flip(); // 切换到读取模式channel.write(buffer); // 写入数据} catch (IOException e) {e.printStackTrace();}// 读取数据try (FileChannel channel = FileChannel.open(Paths.get(filePath), StandardOpenOption.READ)) {ByteBuffer buffer = ByteBuffer.allocate(1024); // 创建缓冲区int bytesRead = channel.read(buffer); // 读取数据if (bytesRead > 0) {buffer.flip(); // 切换到读取模式byte[] data = new byte[buffer.limit()];buffer.get(data); // 从缓冲区获取数据System.out.println(new String(data)); // 打印数据}} catch (IOException e) {e.printStackTrace();}}
}
2.4 使用 Selector
进行网络编程
使用 Selector
监听 TCP 连接:
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;public class SelectorExample {public static void main(String[] args) {int port = 1234;try {ServerSocketChannel serverChannel = ServerSocketChannel.open();serverChannel.socket().bind(new InetSocketAddress(port));serverChannel.configureBlocking(false);Selector selector = Selector.open();serverChannel.register(selector, SelectionKey.OP_ACCEPT);while (true) {int readyChannels = selector.select();if (readyChannels == 0) continue;for (SelectionKey key : selector.selectedKeys()) {if (key.isAcceptable()) {ServerSocketChannel ssc = (ServerSocketChannel) key.channel();SocketChannel clientChannel = ssc.accept();clientChannel.configureBlocking(false);clientChannel.register(selector, SelectionKey.OP_READ);} else if (key.isReadable()) {SocketChannel clientChannel = (SocketChannel) key.channel();ByteBuffer buffer = ByteBuffer.allocate(1024);int bytesRead = clientChannel.read(buffer);if (bytesRead > 0) {buffer.flip();byte[] data = new byte[buffer.limit()];buffer.get(data);System.out.println(new String(data));}}}selector.selectedKeys().clear();}} catch (IOException e) {e.printStackTrace();}}
}
总结
NIO 提供了更高效的数据读写机制,特别是对于大数据量和高并发的应用场景非常有用。通过使用 Buffer
、Channel
和 Selector
,可以实现更灵活的 I/O 操作。在实际应用中,可以根据具体的需求选择适合的 NIO 类来实现高效的数据处理。