使用try-with-resource的輸入輸出流自動(dòng)關(guān)閉
try-with-resource的輸入輸出流自動(dòng)關(guān)閉
最近在做代碼審核的時(shí)候,審核工具提示我將 try-catch-finally 給替換掉,而且根據(jù)公司相關(guān)要求,該提示的級(jí)別還不低,不改不予通過(guò)。
先看看代碼吧:
FileReader fr = null;
BufferedReader br = null;
try {
fr = new FileReader(fileName);
br = new BufferedReader(fr);
return br.readLine();
} catch (Exception e) {
log.error("error:{}", e);
} finally {
if (br != null) {
try {
br.close();
} catch(IOException e){
log.error("error:{}", e);
}
}
if (fr != null ) {
try {
br.close();
} catch(IOException e){
log.error("error:{}", e);
}
}
}
審核工具給出的意見(jiàn)是 替換為:
try (
FileReader fr = new FileReader(fileName);
BufferedReader br = new BufferedReader(fr)
) {
return br.readLine();
}catch (Exception e) {
log.error("error:{}", e);
}
或者是:
try (
BufferedReader br = new BufferedReader(new FileReader(fileName))
) {
// no need to name intermediate resources if you don't want to
return br.readLine();
}
catch (Exception e) {
log.error("error:{}", e);
}
對(duì)比代碼,不難發(fā)現(xiàn),輸入輸出流的關(guān)閉存在著差異。難道輸入輸出流不用關(guān)閉了嗎?
帶著這個(gè)問(wèn)題看看源代碼,發(fā)現(xiàn)
public class FileInputStream extends InputStream{}
public abstract class InputStream implements Closeable {}
/**
* A {@code Closeable} is a source or destination of data that can be closed.
* The close method is invoked to release resources that the object is
* holding (such as open files).
*
* @since 1.5
*/
public interface Closeable extends AutoCloseable {}
/**
* An object that may hold resources (such as file or socket handles)
* until it is closed. The {@link #close()} method of an {@code AutoCloseable}
* object is called automatically when exiting a {@code
* try}-with-resources block for which the object has been declared in
* the resource specification header. This construction ensures prompt
* release, avoiding resource exhaustion exceptions and errors that
* may otherwise occur.
*
* @apiNote
* <p>It is possible, and in fact common, for a base class to
* implement AutoCloseable even though not all of its subclasses or
* instances will hold releasable resources. For code that must operate
* in complete generality, or when it is known that the {@code AutoCloseable}
* instance requires resource release, it is recommended to use {@code
* try}-with-resources constructions. However, when using facilities such as
* {@link java.util.stream.Stream} that support both I/O-based and
* non-I/O-based forms, {@code try}-with-resources blocks are in
* general unnecessary when using non-I/O-based forms.
*
* @author Josh Bloch
* @since 1.7
*/
public interface AutoCloseable {}
AutoCloseable 顧名思義, 自動(dòng)關(guān)閉流. 從注釋中我們可以發(fā)現(xiàn),實(shí)現(xiàn)了AutoCloseable并在try()中聲明的對(duì)象,當(dāng)try-with-resource代碼塊執(zhí)行完的時(shí)候,會(huì)自動(dòng)調(diào)用close()方法。
注意:
一個(gè) try-with-resources 語(yǔ)句可以像普通的 try 語(yǔ)句那樣有 catch 和 finally 塊。在try-with-resources 語(yǔ)句中, 任意的 catch 或者 finally 塊都是在聲明的資源被關(guān)閉以后才運(yùn)行。
使用try-with-resource需要注意的地方
try-with-resource是JDK7引入的語(yǔ)法糖,可以簡(jiǎn)化Autocloseable資源類的關(guān)閉過(guò)程,
比如JDK7以前下面的代碼:
File file = new File("d:/tmp/1.txt");
FileInputStream fis = null;
try {
fis = new FileInputStream(file);
xxxxx
xxxxx
} catch (IOException e) {
e.printStackTrace();
}finally{
if(fis != null){
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
上面是一段讀取文件內(nèi)容的示意代碼,為了防止在try代碼塊中出現(xiàn)異常后導(dǎo)致的資源泄露問(wèn)題,在finally代碼塊中一般處理資源的關(guān)閉事項(xiàng)。
JDK之后上面的代碼就可以簡(jiǎn)化成下面的寫(xiě)法:
File file = new File("d:/tmp/1.txt");
try(FileInputStream fis = new FileInputStream(file);) {
fis.read();
} catch (IOException e) {
e.printStackTrace();
}finally{
}
可以看出是簡(jiǎn)化了不少,之所以稱之為語(yǔ)法糖,是因?yàn)榫幾g成class文件后實(shí)際的代碼就不是這樣的了,編譯過(guò)程中會(huì)自動(dòng)添加資源的關(guān)閉處理。
上面的代碼編譯出的class文件使用javap進(jìn)行反編譯后是下面這樣的
File file = new File("d:/tmp/1.txt");
try {
Throwable var2 = null;
Object var3 = null;
try {
FileInputStream fis = new FileInputStream(file);
xxx
xxxx
} catch (Throwable var12) {
if (var2 == null) {
var2 = var12;
} else if (var2 != var12) {
var2.addSuppressed(var12);
}
throw var2;
}
} catch (IOException var13) {
var13.printStackTrace();
}
好了,上面已經(jīng)引入今天的主題,try-with-resource,但是仍然有需要注意的地方。
比如下面的代碼:
private static class MyResource implements AutoCloseable{
private MyResource1 res;
public MyResource(MyResource1 res){
this.res = res;
}
@Override
public void close() throws Exception {
System.out.println("MyResource自動(dòng)關(guān)閉");
Integer a = null;
a.toString();
this.res.close();
}
}
private static class MyResource1 implements AutoCloseable{
@Override
public void close() throws Exception {
System.out.println("MyResource1自動(dòng)關(guān)閉");
}
}
@Test
public void test() throws Exception{
try(
MyResource r = new MyResource(new MyResource1())){
Integer a = null ;
a.toString();
}
}
執(zhí)行上面的代碼,由于MyResource的close方法中出現(xiàn)了異常,此時(shí)創(chuàng)建的MyResource1就不會(huì)被關(guān)閉,從而出現(xiàn)資源泄露情況,為了規(guī)避這個(gè)問(wèn)題,為了規(guī)避這個(gè)問(wèn)題,我們需要?jiǎng)?chuàng)建的實(shí)現(xiàn)AutoCloseable接口的對(duì)象單獨(dú)創(chuàng)建。
如下面所示:
try(
MyResource1 res= new MyResource1();
MyResource r = new MyResource(res)){
Integer a = null ;
a.toString();
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Logback配置文件這么寫(xiě),還說(shuō)你不會(huì)整理日志?
logback框架會(huì)默認(rèn)加載classpath下命名為logback-spring.xml或logback.xml的配置文件。這篇文章主要介紹了Logback配置文件寫(xiě)法,需要的朋友可以參考下2020-07-07
Spring Cloud Config 使用本地配置文件方式
這篇文章主要介紹了Spring Cloud Config 使用本地配置文件方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Java中關(guān)于優(yōu)先隊(duì)列PriorityQueue的使用及相關(guān)方法
這篇文章主要介紹了Java中關(guān)于優(yōu)先隊(duì)列PriorityQueue的使用及相關(guān)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
Mybatis實(shí)現(xiàn)查詢相冊(cè)數(shù)據(jù)列表流程講解
這篇文章主要介紹了Mybatis實(shí)現(xiàn)查詢相冊(cè)數(shù)據(jù)列表流程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-12-12
Swagger2配置Security授權(quán)認(rèn)證全過(guò)程
這篇文章主要介紹了Swagger2配置Security授權(quán)認(rèn)證全過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03

