如何正确的释放资源?

  Java   5分钟   414浏览   0评论

在Java开发中,我们经常会遇到各种资源的释放问题。

比如最常见的I/O操作,我们往往会通过调用API提供的close方法来关闭流,释放资源。

但是追求极致的程序员会发现,这种方式存在不少问题,比如忘记关闭流、代码不美观、异常不好处理等等。

今天我们就来盘一盘这个问题,最后优雅的去解决掉它。

我们先来看一段代码,大家看看自己有没有中招,顺带认真分析一下这段代码有什么问题!

/**
  * 读取指定路径文件第一行数据内容
  *
  * @param path 文件路径
  * @return
  */
public static String readFirstLine(String path){

    BufferedReader br = null;
    String line = null;
    try {
        br = new BufferedReader(new FileReader(path));
        line = br.readLine();
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if (br != null) {
            try {
                br.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    return line;
}

这段代码实现了一个非常简单的功能,但是为了正确的处理异常,释放资源,导致代码编写得十分臃肿,可读性非常的差!

那我们应该如何优化代码呢?

最好的办法就是采用Java 7引入的try-with-resources语句来解决,当然能使用try-with-resource的资源类,必须实现了AutoCloseable接口。比如我们Demo中的BufferedReader类,它的类关系图如下所示:

try-with-resources使用非常方便,优化后的代码如下所示:

/**
  * 读取指定路径文件第一行数据内容 采用 try-with-resources
  *
  * @param path 文件路径
  * @return
  */
public static String readFirstLine(String path) {

    String line = null;
    try (BufferedReader br = new BufferedReader(new FileReader(path))){
        line = br.readLine();
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    return line;
}

可以看到优化后的代码变得非常清晰,而且不需要手动写代码去释放资源,这样我们就不需要担心资源的关闭问题了!

读到这里有些小伙伴肯定会想,我就是不用try-with-resources,我觉得自己释放才靠谱。这……我只能说你赢了,那有没有一种稍微优雅一点的方式来释放资源呢?

/**
  * 释放资源,需注意资源释放顺序
  *
  * @param closeables
  */
public static void close(Closeable... closeables) {
    if (Objects.nonNull(closeables)) {
        for (Closeable closeable : closeables) {
            if (Objects.nonNull(closeable)) {
                try {
                    closeable.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

我们可以考虑抽取一个方法来释放资源,方法参数是一个可变Closeable参数,我们在手动释放资源的时候一定要注意资源的释放顺序,因此调用方法是参数的顺序一定不能错了!

所以呀,想来想去还是try-with-resources语法香!!!

如果你觉得文章对你有帮助,那就请作者喝杯咖啡吧☕
微信
支付宝
  0 条评论