Java IO: 异常处理

原文链接 作者:Jakob Jenkov 译者: 李璟(jlee381344197@gmail.com)

流与Reader和Writer在结束使用的时候,需要正确地关闭它们。通过调用close()方法可以达到这一点。不过这需要一些思考。请看下边的代码:

InputStream input = new FileInputStream("c:\\data\\input-text.txt");

int data = input.read();

while(data != -1) {

    //do something with data...  

    doSomethingWithData(data);

    data = input.read();

}

input.close();

第一眼看这段代码时,可能觉得没什么问题。可是如果在调用doSomethingWithData()方法时出现了异常,会发生什么呢?没错,这个InputStream对象就不会被关闭。

为了避免异常造成流无法被关闭,我们可以把代码重写成这样:

InputStream input = null;

try{

    input = new FileInputStream("c:\\data\\input-text.txt");

    int data = input.read();

    while(data != -1) {

        //do something with data...

        doSomethingWithData(data);

        data = input.read();

}

}catch(IOException e){

    //do something with e... log, perhaps rethrow etc.

} finally {

    if(input != null)

        input.close();

}

注意到这里把InputStream的关闭代码放到了finally块中,无论在try-catch块中发生了什么,finally内的代码始终会被执行,所以这个InputStream总是会被关闭。

但是如果close()方法抛出了异常,告诉你流已经被关闭过了呢?为了解决这个难题,你也需要把close()方法写在try-catch内部,就像这样:

} finally {

    try{

        if(input != null)

            input.close();

    } catch(IOException e){

        //do something, or ignore.

    }
}

这段解决了InputStream(或者OutputStream)流关闭的问题的代码,确实是有一些不优雅,尽管能够正确处理异常。如果你的代码中重复地遍布了这段丑陋的异常处理代码,这不是很好的一个解决方案。如果一个匆忙的家伙贪图方便忽略了异常处理呢?

此外,想象一下某个异常最先从doSomethingWithData方法内抛出。第一个catch会捕获到异常,然后在finally里程序会尝试关闭InputStream。但是如果还有异常从close()方法内抛出呢?这两个异常中得哪个异常应当往调用栈上传播呢?

幸运的是,有一个办法能够解决这个问题。这个解决方案称作“异常处理模板”。创建一个正确关闭流的模板,能够在代码中做到一次编写,重复使用,既优雅又简单。详情参见Java异常处理模板

Java7中IO的异常处理

从Java7开始,一种新的被称作“try-with-resource”的异常处理机制被引入进来。这种机制旨在解决针对InputStream和OutputStream这类在使用完毕之后需要关闭的资源的异常处理。可以浏览Try with Resource in Java 7获得更多信息。 

时间: 2024-08-31 09:43:52

Java IO: 异常处理的相关文章

Java IO: OutputStream

原文链接 作者: Jakob Jenkov 译者: 李璟(jlee381344197@gmail.com) OutputStream类是Java IO API中所有输出流的基类.子类包括BufferedOutputStream,FileOutputStream等等.参考Java IO概述这一小节底部的表格,可以浏览完整的子类的列表. 输出流和目标媒介 输出流往往和某些数据的目标媒介相关联,比如文件,网络连接,管道等.更多细节请参考Java IO概述.当写入到输出流的数据逐渐输出完毕时,目标媒介是

Java IO教程

Java IO 是一套Java用来读写数据(输入和输出)的API.大部分程序都要处理一些输入,并由输入产生一些输出.Java为此提供了java.io包. 如果你浏览下java.io包,会对其中各样的类选择感到迷惑.这些类的作用都是什么?对于某个任务该选择哪个类?怎样创建你自己的类做插件?这个手册的目的就是给你介绍这些类是如何组织的,以及怎样使用他们,因此你就不会疑惑需要时怎样选取合适的类,或者是否有一个满足你需求的类已经存在了. Java.io 包的范围 java.io 包并没有涵盖所有输入输出

Java IO: InputStream

原文链接 作者: Jakob Jenkov 译者: 李璟(jlee381344197@gmail.com) InputStream类是Java IO API中所有输入流的基类.InputStream子类包括FileInputStream,BufferedInputStream,PushbackInputStream等等.参考Java IO概述这一小节底部的表格,可以浏览完整的InputStream子类的列表. Java InputStream例子 InputStream用于读取基于字节的数据,一

Java IO: ByteArray和Filter

本小节会简要概括Java IO中字节数组与过滤器的输入输出流,主要涉及以下4个类型的流:ByteArrayInputStream,ByteArrayOutputStream,FilterInputStream,FilterOutputStream.请注意,为了清晰,这里忽略了必要的异常处理.想了解更多异常处理的信息,请参考Java IO异常处理. ByteArrayInputStream 原文链接 ByteArrayInputStream允许你从字节数组中读取字节流数据,代码如下: 01 byt

Java IO: FileReader和FileWriter

本章节将简要介绍FileReader和FileWriter.与FileInputStream和FileOutputStream类似,FileReader与FileWriter用于处理文件内容. FileReader 原文链接 FileReader能够以字符流的形式读取文件内容.除了读取的单位不同之外(译者注:FileReader读取字符,FileInputStream读取字节),FileReader与FileInputStream并无太大差异,也就是说,FileReader用于读取文本.根据不同

Java IO: RandomAccessFile

RandomAccessFile允许你来回读写文件,也可以替换文件中的某些部分.FileInputStream和FileOutputStream没有这样的功能. 创建一个RandomAccessFile 在使用RandomAccessFile之前,必须初始化它.这是例子: 1 RandomAccessFile file = new RandomAccessFile("c:\\data\\file.txt", "rw"); 请注意构造函数的第二个参数:"rw

Java IO: InputStreamReader和OutputStreamWriter

作者: Jakob Jenkov 译者: 李璟(jlee381344197@gmail.com) 本章节将简要介绍InputStreamReader和OutputStreamWriter.细心的读者可能会发现,在之前的文章中,IO中的类要么以Stream结尾,要么以Reader或者Writer结尾,那这两个同时以字节流和字符流的类名后缀结尾的类是什么用途呢?简单来说,这两个类把字节流转换成字符流,中间做了数据的转换,类似适配器模式的思想. InputStreamReader 原文链接 Input

Java IO: 字符流的Piped和CharArray

本章节将简要介绍管道与字符数组相关的reader和writer,主要涉及PipedReader.PipedWriter.CharArrayReader.CharArrayWriter. PipedReader 原文链接 PipedReader能够从管道中读取字符流.与PipedInputStream类似,不同的是PipedReader读取的是字符而非字节.换句话说,PipedReader用于读取管道中的文本.代码如下: Reader reader = new PipedReader(pipedW

Java IO: PipedOutputStream

PipedOutputStream可以往管道里写入读取字节流数据,代码如下: OutputStream output = new PipedOutputStream(pipedInputStream); while(moreData) { int data = getMoreData(); output.write(data); } output.close(); 请注意,为了清晰,这里忽略了必要的异常处理.想了解更多异常处理的信息,请参考Java IO异常处理. PipedOutputStre