跳到主要内容Java IO 流概述与基础应用 | 极客日志Javajava
Java IO 流概述与基础应用
综述由AI生成Java IO 流是处理数据输入输出的核心机制。 IO 流的分类(字节流/字符流、节点流/处理流)及超类结构。内容涵盖文件流读写、块读写优化、缓冲流应用、对象序列化与反序列化,以及字符转换流的使用场景和编码处理。通过代码示例演示了文件复制、文本读写及对象持久化的基本实践,帮助开发者掌握 Java IO 的核心用法。
ArchDesign6K 浏览 IO 流简介
IO 是什么
Java 中的 IO 流是用于处理数据输入和输出的核心机制。通过应用 IO 流可以使 Java 程序能够与外部世界(如磁盘文件、网络、硬件设备等)进行数据交互。IO 流的全称为输入/输出流(Input/Output Stream),它是 Java 编程语言中用于数据传输的一种抽象模型。流可以被想象为数据的连续流动,就像水通过管道一样,数据通过流从一个地方流向另一个地方。
应用场景
IO 流在 Java 开发中几乎无处不在,任何涉及到数据读写的地方都会用到 IO 流。常见的应用场景包括文件读写、网络通信、数据持久化等。
IO 流的分类
IO 流的分类可以从多个维度进行理解:
- 按流的方向:可以分为输入流 (InputStream) 和输出流 (OutputStream)
- 按流的功能:可以分为节点流和处理流
- 按数据处理单元:可以分为字节流 (以字节读写为基本单位) 和字符流 (以字符读写为基本单位)
流的超类类型
在 Java 中,输入输出 (IO) 流的超类主要有两类,分别对应于字节流和字符流。
-
字节流 (Byte Streams)
- java.io.InputStream:所有字节输入流的超类。定义了从源读取字节的基本方法
- java.io.OutputStream:所有字节输出流的超类。定义了向目的地写入字节的基本方法
-
字符流 (Character Streams)
- java.io.Reader:字符输入流的超类,定义了从源读取字符的基本方法
- java.io.Writer:字符输出流的超类,定义了向目的地写入字符的基本方法
这些超类提供了基本的读写操作,而它们的子类则实现了特定的读写功能,例如从文件读取、向网络套接字写入、数据的缓冲等。例如:
- 字节流的具体实现包括:
- FileInputStream 和 FileOutputStream:用于读写文件。
- BufferedInputStream 和 BufferedOutputStream:提供缓冲,从而提高读写效率。
- ObjectInputStream 和 ObjectOutputStream:用于对象的序列化和反序列化。
- 字符流的具体实现包括:
- FileReader 和 FileWriter:用于读写文本文件。
- BufferedReader 和 BufferedWriter、PrintWriter:提供字符缓冲,提高读写效率。
- InputStreamReader 和 OutputStreamWriter:用于在字节流和字符流之间转换。
这些类构成了 Java IO 流的层次结构,允许开发者根据不同的需求选择合适的流类型来进行数据的读写操作。
字节文件流应用
简介
文件流是用来连接我们的程序与文件之间的'管道',用来读写文件中的数据。
核心 API
文件流是继承自 InputStream 和 OutputStream 的流对象,其分类为:
- 文件输入流 java.io.FileInputStream:读取文件数据的流
- 文件输出流 java.io.FileOutputStream:写入文件数据的流
文件输出流应用
java.io.FileOutputStream 输出流对象用于向文件中写入数据,对象构建通常会借助如下两个构造方法:
FileOutputStream(String path)
创建文件输出流对指定的 path 路径表示的文件进行写操作,如果该文件不存在则将其创建出来
FileOutputStream(File file)
创建文件输出流对指定的 file 对象表示的文件进行写操作,如果该文件不存在则将其创建出来
案例
io;
java.io.File;
java.io.FileNotFoundException;
java.io.FileOutputStream;
java.io.IOException;
{
IOException {
();
file.getParentFile();
System.out.println(parent);
(!parent.exists()) {
parent.mkdirs();
System.out.println();
}
(file);
fos2.write();
fos2.write();
fos2.write();
System.out.println();
fos2.close();
}
}
package
import
import
import
import
public
class
FOSDemo01
public
static
void
main
(String[] args)
throws
File
file
=
new
File
"jsd/2406/fos.dat"
File
parent
=
if
"文件不存在,创建成功"
FileOutputStream
fos2
=
new
FileOutputStream
3
4
5
"数据写入 OK"
文件输入流应用
java.io.FileInputStream 文件输入流对象用于从文件中读取数据,常用构造方法有:
FileInputStream(String path)
基于给定的路径对应的文件创建文件输入流
FileInputStream(File file)
基于给定的 File 对象所表示的文件创建文件输入流
package io;
import java.io.FileInputStream;
import java.io.IOException;
public class FISDemo01 {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("jsd/2406/fos.dat");
int n = 0;
while ((n = fis.read()) != -1) {
System.out.println(n);
}
fis.close();
}
}
文件的复制实践
复制文件的原理就是使用文件输入流从原文件中陆续读取出每一个字节,然后再使用文件输出流将字节陆续写入到另一个文件中完成的。
package io;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class FileCopyDemo01 {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("a.png");
FileOutputStream fos = new FileOutputStream("b.png");
int n = -1;
while ((n = fis.read()) != -1) {
fos.write(n);
}
fis.close();
fos.close();
}
}
块读写操作实践
- 块读:一次性读取一组字节的方式
InputStream 中定义了块读的方法
int read(byte[] data)
一次性读取给定字节数组总长度的字节量并存入到该数组中。
返回值为实际读取到的字节数。如果返回值为 -1 表示本次没有读取到任何字节已经是流的末尾了
- 块写:一次性写出一组字节的方式
OutputStream 中定义了块写的方法
void write(byte[] data)
一次性将给定数组中所有字节写出
void write(byte[] data, int offset, int len)
一次性将 data 数组中从下标 offset 处开始的连续 len 个字节一次性写出
package io;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class FileCopyDemo02 {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("a.png");
FileOutputStream fos = new FileOutputStream("b.png");
byte[] data = new byte[1024];
int length = -1;
while ((length = fis.read(data)) != -1) {
fos.write(data, 0, length);
}
fis.close();
fos.close();
}
}
文本数据操作
对于 FileInputStream 和 FileOutputStream 提供的方法只能读写字节数据,对于文本数据如何操作呢?
package io;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
public class WriteStringDemo01 {
public static void main(String[] args) throws IOException {
String str = "我爱 Java";
FileOutputStream fos = new FileOutputStream("./f1.txt");
byte[] data = str.getBytes(StandardCharsets.UTF_8);
fos.write(data);
str = "Hello Tedu";
fos.write(str.getBytes(StandardCharsets.UTF_8));
fos.close();
}
}
package io;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
public class ReadStringDemo01 {
public static void main(String[] args) throws IOException {
FileInputStream fis = new FileInputStream("./f1.txt");
byte[] data = new byte[fis.available()];
int len = fis.read(data);
String str = new String(data, 0, len);
System.out.println(str);
fis.close();
}
}
字节缓冲流应用
处理流
概述
处理流是 Java IO 流体系中的一类重要的流,它们是在已经存在的流(称为基础流或节点流)之上构建的,为这些流添加额外的功能,而不改变原流本身。处理流的主要目的是为了增强或改变流的行为,比如提供缓冲、转换数据类型、序列化等等。
应用特征
处理流的典型特征是它们自身并不直接连接到物理的输入/输出设备,而是'装饰'在其他流之上,利用这些流进行数据的读写,并在此基础上提供附加服务。处理流的实例通常会把读写操作委托给底层的流,同时可能对数据进行一些预处理或后处理。
处理流可以串联使用,即一个处理流的输出可以作为另一个处理流的输入,形成流的链式结构。例如,可以先使用 BufferedInputStream 对 FileInputStream 进行包装,然后再用其它处理流包装这个缓冲流。
字节缓冲流
Java 中的缓冲流是一种处理流,有 java.io.BufferedInputStream 和 BufferedOutputStream,其作用就是加快读写效率,通常缓冲流的应用最终要链接在低级流上。
- 缓冲字节输出流
BufferedOutputStream(OutputStream out)
实例化一个缓冲字节输出流并链接在指定的字节输出流上。默认缓冲区大小 8kb(内部维护的 byte[] buf 数组长度 8192)
BufferedOutputStream(OutputStream out, int size)
实例化一个指定缓冲区大小的缓冲字节输出流并链接在指定的字节输出流上。
- 缓冲字节输入流
BufferedInputStream(InputStream in)
实例化一个缓冲字节输入流并链接在指定的字节输入流上。默认缓冲区大小 8kb(内部维护的 byte[] buf 数组长度 8192)
BufferedInputStream(InputStream in, int size)
实例化一个指定缓冲区大小的缓冲字节输入流并链接在指定的字节输入流上。
应用案例分析
package io;
import java.io.*;
public class FileCopyDemo03 {
public static void main(String[] args) throws IOException {
BufferedInputStream bis = new BufferedInputStream(
new FileInputStream("./a.png")
);
BufferedOutputStream bos = new BufferedOutputStream(
new FileOutputStream("./c.png")
);
byte[] data = new byte[16];
int len;
while ((len = bis.read(data)) != -1) {
bos.write(data, 0, len);
}
bos.flush();
bis.close();
bos.close();
}
}
对象流应用实践
简介
Java 中的对象流也是一种处理流,有 java.io.ObjectInputStream 和 java.io.ObjectOutputStream 两个类型,用于实现对象的读写。也就是对象的序列化(把对象转化为字节)和反序列化(把字节转换为对象)。
- ObjectOutputStream:用于将 Java 对象序列化为字节流,以便可以将对象写入文件、网络或其他任何形式的持久存储中。
- ObjectInputStream:用于从字节流中反序列化 Java 对象,即将字节流重新转换为 Java 对象。
Serializable 接口
为了能够被序列化,一个对象所属的类必须实现 Serializable 接口。Serializable 是一个标记接口,不包含任何方法,它的存在仅仅是为了标识一个类是可序列化的。如果尝试序列化一个没有实现 Serializable 接口的类的对象,将会抛出 NotSerializableException 异常。
应用案例分析
package io;
import java.io.Serializable;
public class Message implements Serializable {
private static final long serialVersionUID = 8725255648785219540L;
private Integer id;
private transient String title;
private String content;
public Message() {}
public Message(Integer id, String title, String content) {
this.id = id;
this.title = title;
this.content = content;
}
public Integer getId() { return id; }
public void setId(Integer id) { this.id = id; }
public String getTitle() { return title; }
public void setTitle(String title) { this.title = title; }
public String getContent() { return content; }
public void setContent(String content) { this.content = content; }
@Override
public String toString() {
return "Message{" +
"id=" + id +
", title='" + title + '\'' +
", content='" + content + '\'' +
'}';
}
}
package io;
import java.io.BufferedOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
public class ObjectOutputStreamDemo01 {
public static void main(String[] args) throws IOException {
Message m1 = new Message(1, "传奇老师病了", "做了个小手术");
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("message.txt"));
oos.writeObject(m1);
System.out.println("序列化 OK");
oos.close();
}
}
package io;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
public class ObjectInputStreamDemo02 {
public static void main(String[] args) throws IOException, ClassNotFoundException {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("message.txt"));
Message msg = (Message) ois.readObject();
System.out.println(msg);
ois.close();
}
}
字符转换流应用
字符流
在 Java 中,字符流(Character Streams)是用于处理文本数据的一种流类型,其基类是 java.io.Reader 和 java.io.Writer 两个抽象类。
应用场景?
字符流广泛应用于读写文本文件,处理字符串,以及在网络通信中传输文本数据。由于文本文件通常包含人类可读的内容,字符流可以确保正确处理各种字符集和编码,从而避免乱码问题。
- 编码透明性:字符流允许你指定字符编码,因此你可以读写使用不同编码的文本文件,而不需要了解具体的编码细节。
- 字符边界处理:当处理多字节编码(如 UTF-8)的字符时,字符流能确保每次读写的都是一个完整的字符,避免字节流中可能出现的字符截断问题(这里其实是乱码)
字符转换流应用
在 Java 中,字符转换流是 Java I/O 体系中的一类流,它们实现了字节流和字符流之间的转换。这类流包括 InputStreamReader 和 OutputStreamWriter 两个核心类。
InputStreamReader 是一个读取字节并使用指定的字符集将其解码为字符的 Reader;OutputStreamWriter 则是一个接收字符并使用指定的字符集将其编码为字节的 Writer。
- 当你需要从字节流中读取文本数据,并且想要控制使用的字符编码时。
- 当你需要将字符数据写入到字节流中,并指定字符编码时。
- 在网络通信中,当你发送或接收的数据是以字节形式存在,但你希望以字符形式处理时。
- 编码兼容性:不同的系统和文件可能使用不同的字符编码,字符转换流可以让你指定编码,从而正确解析或生成文本。
- 国际化支持:为了支持多种语言和字符集,字符转换流提供了灵活的编码选择,这对于开发国际化应用非常重要。
- 数据完整性:在处理多字节编码的字符时,字节流可能会导致字符被错误地分割,而字符转换流保证了字符的完整性和正确性。
应用案例分析和实践
具体应用:通过字符转换流将一些字符串写入到指定文件
package io;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.nio.charset.StandardCharsets;
public class OutputStreamWriterDemo01 {
public static void main(String[] args) throws IOException {
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("./osw.txt"), StandardCharsets.UTF_8);
osw.write("你好");
osw.write("子千老师");
System.out.println("数据写出完毕");
osw.close();
}
}
package io;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
public class InputStreamReaderDemo01 {
public static void main(String[] args) throws IOException {
InputStreamReader isr = new InputStreamReader(new FileInputStream("./osw.txt"), StandardCharsets.UTF_8);
int data = 0;
while ((data = isr.read()) != -1) {
System.out.print((char) data);
}
isr.close();
}
}
字符缓冲流应用
简介
在 Java 中,字符缓冲流是用于提高读写性能的一对流,它们包括 BufferedReader 和 BufferedWriter。这些流通过内部缓存机制减少了对底层 I/O 设备的访问频率,从而提高了读写操作的效率。
在字符输出缓冲流应用时,通常会结合 PrintWriter 一起使用。这个对象可以一次写一行,并且可以自带刷新机制。
快速入门实现
案例 1:PrintWriter 不同构造方法应用
package io;
import java.io.*;
public class PrintWriterDemo01 {
public static void main(String[] args) throws FileNotFoundException {
PrintWriter pw = new PrintWriter("./pw.txt");
pw.println("hello");
pw.println("world");
pw.println("Java");
pw.close();
}
}
package io;
import java.io.*;
public class PrintWriterDemo02 {
public static void main(String[] args) throws FileNotFoundException {
PrintWriter printWriter = new PrintWriter(
new BufferedWriter(
new OutputStreamWriter(new FileOutputStream("./pw.txt"))
), true
);
printWriter.println("hello");
printWriter.println("Java");
printWriter.close();
}
}
package io;
import java.io.*;
import java.nio.charset.StandardCharsets;
public class BufferedReaderDemo01 {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(
new InputStreamReader(
new FileInputStream("./src/main/java/io/FISDemo01.java"),
StandardCharsets.UTF_8
)
);
String line = null;
while ((line = br.readLine()) != null) {
System.out.println(line);
}
br.close();
}
}
相关免费在线工具
- Keycode 信息
查找任何按下的键的javascript键代码、代码、位置和修饰符。 在线工具,Keycode 信息在线工具,online
- Escape 与 Native 编解码
JavaScript 字符串转义/反转义;Java 风格 \uXXXX(Native2Ascii)编码与解码。 在线工具,Escape 与 Native 编解码在线工具,online
- JavaScript / HTML 格式化
使用 Prettier 在浏览器内格式化 JavaScript 或 HTML 片段。 在线工具,JavaScript / HTML 格式化在线工具,online
- JavaScript 压缩与混淆
Terser 压缩、变量名混淆,或 javascript-obfuscator 高强度混淆(体积会增大)。 在线工具,JavaScript 压缩与混淆在线工具,online
- Base64 字符串编码/解码
将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online
- Base64 文件转换器
将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online