该Java教程是针对JDK 8编写的。本页面描述的示例和实践不利用后续版本引入的改进,并且可能使用不再可用的技术。
请参阅Java语言变更,了解Java SE 9及其后续版本中更新的语言功能的摘要。
请参阅JDK发行说明,了解有关所有JDK版本的新功能、增强功能和已删除或弃用选项的信息。
程序使用字节流来执行8位字节的输入和输出。所有字节流类都是从InputStream
和OutputStream
继承而来。
有许多字节流类。为了演示字节流的工作原理,我们将重点介绍文件I/O字节流,即FileInputStream
和FileOutputStream
。其他类型的字节流使用方式基本相同,只是构造方式有所不同。
我们将通过检查一个名为CopyBytes
的示例程序来探索FileInputStream
和FileOutputStream
,该程序使用字节流以每次一个字节的方式复制xanadu.txt
。
import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; public class CopyBytes { public static void main(String[] args) throws IOException { FileInputStream in = null; FileOutputStream out = null; try { in = new FileInputStream("xanadu.txt"); out = new FileOutputStream("outagain.txt"); int c; while ((c = in.read()) != -1) { out.write(c); } } finally { if (in != null) { in.close(); } if (out != null) { out.close(); } } } }
CopyBytes
大部分时间都在一个简单的循环中,该循环以每次一个字节的方式读取输入流并写入输出流,如下图所示。
简单的字节流输入和输出。
在不再需要流时关闭它是非常重要的,因此CopyBytes
使用finally
块来保证即使发生错误,两个流也将被关闭。这种做法有助于避免严重的资源泄漏。
可能的一个错误是CopyBytes
无法打开一个或两个文件。当发生这种情况时,对应于文件的流变量从其初始的null
值不会改变。这就是为什么CopyBytes
确保每个流变量在调用close
之前包含一个对象引用的原因。
CopyBytes
看起来像是一个正常的程序,但实际上它代表了一种应该避免使用的低级I/O。由于xanadu.txt
包含字符数据,最好的方法是使用字符流,如下一节所讨论的。还有更复杂数据类型的流。字节流只应用于最基本的I/O。
那么为什么要讨论字节流呢?因为所有其他流类型都是基于字节流构建的。