吃透Java IO!从字节流到字符流,一篇讲明白
2026/5/16 22:46:21 网站建设 项目流程

作为Java程序员,不管是读写文件、处理网络数据,还是操作控制台输入输出,IO 都是绕不开的坎。刚学的时候,看着 InputStream、OutputStream、Reader、Writer 一堆类,属实有点头大。今天就用大白话,把Java IO的核心知识点捋清楚,新手也能轻松看懂!

一、先搞懂:Java IO到底是啥?

IO,全称 Input/Output,翻译过来就是输入和输出。说白了,就是程序和外部设备之间的数据传输。

比如你用Java程序读取本地的txt文件,这就是输入(Input) —— 数据从文件进到程序里;你把程序里的用户信息写入到数据库,或者保存成一个新文件,这就是输出(Output) —— 数据从程序跑到外部存储里。

Java把这些IO操作都封装在了 java.io 包下,后来又出了 java.nio (NIO是Non-blocking IO,非阻塞IO,今天先聊传统IO)。核心思路就是用流的方式处理数据,这“流”就像水管,数据就是水管里的水,顺着水管就能从一端流到另一端。

二、核心分类:字节流 vs 字符流

Java IO最核心的划分,就是字节流和字符流。这俩的区别,直接决定了你该用哪个类干活。

2.1 字节流:处理一切数据的“万能选手”

字节流的操作单位是 字节(byte),1个字节等于8位。不管是文本文件、图片、音频、视频,本质上都是字节组成的,所以字节流能处理任何类型的数据。

字节流的顶级父类是两个抽象类:

- InputStream:所有字节输入流的爹,负责读数据

- OutputStream:所有字节输出流的爹,负责写数据

我们常用的是它们的子类,比如操作文件的 FileInputStream 和 FileOutputStream。

举个简单的例子:用字节流复制一张图片

java

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

public class ByteStreamDemo {

public static void main(String[] args) {

// 源文件路径和目标文件路径

String srcPath = "D:/test.jpg";

String destPath = "D:/test_copy.jpg";

// 声明流对象

FileInputStream fis = null;

FileOutputStream fos = null;

try {

// 创建流对象

fis = new FileInputStream(srcPath);

fos = new FileOutputStream(destPath);

// 定义缓冲区,每次读1024字节,提高效率

byte[] buffer = new byte[1024];

int len; // 记录每次实际读取的字节数

// 循环读取:读到末尾时,fis.read()会返回-1

while ((len = fis.read(buffer)) != -1) {

// 把读到的字节写入目标文件

fos.write(buffer, 0, len);

}

System.out.println("图片复制成功!");

} catch (IOException e) {

e.printStackTrace();

} finally {

// 关闭流,释放资源,顺序是先开后关

try {

if (fos != null) fos.close();

if (fis != null) fis.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

}

这里要注意两个点:一是一定要用 finally 关闭流,不然会浪费系统资源;二是用字节数组当缓冲区,比一次读一个字节快太多了!

2.2 字符流:处理文本的“专业选手”

字符流的操作单位是 字符(char),它是为了处理文本数据而生的。因为不同的编码格式(比如UTF-8、GBK),一个字符对应的字节数不一样,字符流会帮我们自动处理编码问题,避免出现乱码。

字符流的顶级父类也是两个抽象类:

- Reader:所有字符输入流的爹,负责读文本

- Writer:所有字符输出流的爹,负责写文本

常用子类比如 FileReader 和 FileWriter,直接用来读写文本文件超方便。

同样举个例子:用字符流读取txt文件内容

java

import java.io.FileReader;

import java.io.IOException;

public class CharStreamDemo {

public static void main(String[] args) {

String filePath = "D:/test.txt";

FileReader fr = null;

try {

fr = new FileReader(filePath);

char[] buffer = new char[1024];

int len;

while ((len = fr.read(buffer)) != -1) {

// 把字符数组转成字符串输出

System.out.print(new String(buffer, 0, len));

}

} catch (IOException e) {

e.printStackTrace();

} finally {

try {

if (fr != null) fr.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

}

这里要注意:字符流不能处理图片、音频等二进制文件,强行用的话,文件会损坏!

三、进阶技巧:缓冲流,让IO速度飞起来

刚才的例子里,我们自己定义了字节数组/字符数组当缓冲区,但Java其实给我们提供了更方便的缓冲流,它的底层就是自带了缓冲区,能大大减少磁盘的读写次数,提升效率。

缓冲流分为字节缓冲流和字符缓冲流:

- 字节缓冲流: BufferedInputStream 、 BufferedOutputStream

- 字符缓冲流: BufferedReader 、 BufferedWriter

尤其是字符缓冲流,还提供了 readLine() (按行读)和 newLine() (换行)方法,处理文本简直不要太爽!

举个字符缓冲流的例子:按行读取文本并写入新文件

java

import java.io.BufferedReader;

import java.io.BufferedWriter;

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

public class BufferedStreamDemo {

public static void main(String[] args) {

String srcPath = "D:/source.txt";

String destPath = "D:/dest.txt";

BufferedReader br = null;

BufferedWriter bw = null;

try {

br = new BufferedReader(new FileReader(srcPath));

bw = new BufferedWriter(new FileWriter(destPath));

String line;

// 按行读取,读到末尾返回null

while ((line = br.readLine()) != null) {

bw.write(line);

bw.newLine(); // 写入换行符,不然所有内容会挤在一行

}

System.out.println("文本复制完成!");

} catch (IOException e) {

e.printStackTrace();

} finally {

try {

if (bw != null) bw.close();

if (br != null) br.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

}

四、必踩的坑:这些注意事项要记牢

1. 流一定要关闭:不管是手动关还是用try-with-resources语法(JDK7及以上支持),不关闭流会导致资源泄漏。try-with-resources会自动关闭流,推荐使用!

java

// try-with-resources语法,流对象写在try的括号里

try (FileInputStream fis = new FileInputStream("test.jpg");

FileOutputStream fos = new FileOutputStream("test_copy.jpg")) {

// 读写操作

} catch (IOException e) {

e.printStackTrace();

}

2. 区分绝对路径和相对路径:绝对路径是从盘符开始的完整路径(比如D:/test.txt),相对路径是相对于项目根目录的路径(比如src/test.txt),别搞混了导致文件找不到。

3. 处理编码问题:如果用字符流读写出现乱码,可以指定编码格式,比如用 InputStreamReader 包装字节流:

java

BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("test.txt"), "UTF-8"));

4. 字节流和字符流别混用:比如用FileOutputStream写文本,再用FileReader读,很容易出现乱码。

五、总结

Java IO其实没那么复杂,记住核心逻辑就行:

- 按数据类型分:字节流处理一切数据,字符流专门处理文本

- 按功能分:节点流(直接操作文件/设备,比如FileInputStream)和处理流(包装节点流,比如BufferedInputStream,提升性能)

- 关键操作:打开流→读写数据→关闭流

掌握这些基础,再去学NIO、NIO2就会轻松很多。希望这篇文章能帮到正在啃Java IO的小伙伴们,祝大家编程愉快!

需要专业的网站建设服务?

联系我们获取免费的网站建设咨询和方案报价,让我们帮助您实现业务目标

立即咨询