Java程序中如何正确的释放资源

简介: 在Java开发中,程序员经常会遇到各种资源的释放问题。比如最常见的I/O操作,我们往往会通过调用API提供的close方法来关闭流,释放资源。但是追求极致的程序员会发现,这种方式存在不少问题,比如忘记关闭流、代码不美观、异常不好处理等等。

简介

在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语法香!!!

目录
相关文章
|
1月前
|
Java
java程序导出堆文件
java程序导出堆文件
|
1月前
|
SQL Oracle Java
sql文件批处理程序-java桌面应用
sql文件批处理程序-java桌面应用
25 0
|
1月前
|
存储 Java
Java:编写程序,计算两个数的和、差、积、商和余数。docx
Java:编写程序,计算两个数的和、差、积、商和余数。docx
|
1月前
|
安全 Java 数据库连接
【Java每日一题】——第四十一题:编写程序描述影视歌三栖艺人。
【Java每日一题】——第四十一题:编写程序描述影视歌三栖艺人。
32 0
|
1月前
|
Java 程序员 编译器
【详识JAVA语言】面向对象程序三大特性之二:继承
【详识JAVA语言】面向对象程序三大特性之二:继承
47 2
|
15天前
|
Java Maven
【Java报错】显示错误“Error:java: 程序包org.springframework.boot不存在“
【Java报错】显示错误“Error:java: 程序包org.springframework.boot不存在“
35 3
|
2天前
|
Java
网页运行java程序cheerpj
网页运行java程序cheerpj
27 0
|
30天前
|
Java
elasticsearch使用java程序添加删除修改
elasticsearch使用java程序添加删除修改
9 0
|
1月前
|
Java
java程序
re是java运行时的环境,包含jvm和运行时所需要的类库 jdk是java开的程序包,包含jre和开发人员使用的工具 jvm就是我们常说的java虚拟机,他是整个java实现跨平台的最核心 的部分,所有的java程序会首先被编译为.class的类文件,这种类文 件可以在虚拟机上执行。也就是说class并不直接与机器的操作系统 相对应,而是经过虚拟机间接与操作系统交互,由虚拟机将程序解释 给本地系统执行。 只有jvm还不能成class的执行,因为再解释class的时候jvm需要调用 解释所需要的类库lib,而jre包含lib类库。jvm屏蔽了与具体操作系 统平台相关的信息,使得java程
17 0
|
1月前
|
存储 Java Shell
Java程序结构
Java程序结构
12 3