溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java如何優(yōu)雅地關(guān)閉資源try-with-resource及其異常抑制

發(fā)布時(shí)間:2020-08-29 11:35:40 來源:腳本之家 閱讀:121 作者:敲代碼的小阿貍 欄目:編程語(yǔ)言

一、背景

我們知道,在Java編程過程中,如果打開了外部資源(文件、數(shù)據(jù)庫(kù)連接、網(wǎng)絡(luò)連接等),我們必須在這些外部資源使用完畢后,手動(dòng)關(guān)閉它們。因?yàn)橥獠抠Y源不由JVM管理,無(wú)法享用JVM的垃圾回收機(jī)制,如果我們不在編程時(shí)確保在正確的時(shí)機(jī)關(guān)閉外部資源,就會(huì)導(dǎo)致外部資源泄露,緊接著就會(huì)出現(xiàn)文件被異常占用,數(shù)據(jù)庫(kù)連接過多導(dǎo)致連接池溢出等諸多很嚴(yán)重的問題。

 二、傳統(tǒng)的資源關(guān)閉方式

為了確保外部資源一定要被關(guān)閉,通常關(guān)閉代碼被寫入finally代碼塊中,當(dāng)然我們還必須注意到關(guān)閉資源時(shí)可能拋出的異常,于是變有了下面的經(jīng)典代碼:

public static void main(String[] args) {
  FileInputStream inputStream = null;
  try {
    inputStream = new FileInputStream(new File("test"));
    System.out.println(inputStream.read());
  } catch (IOException e) {
    throw new RuntimeException(e.getMessage(), e);
  } finally {
    if (inputStream != null) {
      try {
        inputStream.close();
      } catch (IOException e) {
        throw new RuntimeException(e.getMessage(), e);
      }
    }
  }
}

熟悉其他語(yǔ)言的朋友可能會(huì)開始吐槽了,在C++中,我們可以把關(guān)閉資源的代碼放在析構(gòu)函數(shù)中,在C#中,我們有using代碼塊。這些語(yǔ)法都有一個(gè)共同的特性,讓外部資源的關(guān)閉行為與外部資源的句柄對(duì)象的生命周期關(guān)聯(lián),當(dāng)外部資源的句柄對(duì)象生命周期終結(jié)時(shí)(例如句柄對(duì)象已出作用域),外部資源的關(guān)閉行為將被自動(dòng)調(diào)用。這樣不僅更加符合面向?qū)ο蟮木幊汤砟睿▽㈥P(guān)閉外部資源的行為內(nèi)聚在外部資源的句柄對(duì)象中),也讓代碼更加簡(jiǎn)潔易懂。怎么到了Java這里,就找不到自動(dòng)關(guān)閉外部資源的語(yǔ)法特性了呢。

三、JDK7及其之后的資源關(guān)閉方式

3.1 try-with-resource語(yǔ)法

確實(shí),在JDK7以前,Java沒有自動(dòng)關(guān)閉外部資源的語(yǔ)法特性,直到JDK7中新增了try-with-resource語(yǔ)法,才實(shí)現(xiàn)了這一功能。

那什么是try-with-resource呢?簡(jiǎn)而言之,當(dāng)一個(gè)外部資源的句柄對(duì)象(比如FileInputStream對(duì)象)實(shí)現(xiàn)了AutoCloseable接口,那么就可以將上面的板式代碼簡(jiǎn)化為如下形式:

public static void main(String[] args) {
  try (FileInputStream inputStream = new FileInputStream(new File("test"))) {
    System.out.println(inputStream.read());
  } catch (IOException e) {
    throw new RuntimeException(e.getMessage(), e);
  }
}

將外部資源的句柄對(duì)象的創(chuàng)建放在try關(guān)鍵字后面的括號(hào)中,當(dāng)這個(gè)try-catch代碼塊執(zhí)行完畢后,Java會(huì)確保外部資源的close方法被調(diào)用。代碼是不是瞬間簡(jiǎn)潔許多!

3.2 實(shí)現(xiàn)原理

try-with-resource并不是JVM虛擬機(jī)的新增功能,只是JDK實(shí)現(xiàn)了一個(gè)語(yǔ)法糖,當(dāng)你將上面代碼反編譯后會(huì)發(fā)現(xiàn),其實(shí)對(duì)JVM虛擬機(jī)而言,它看到的依然是之前的寫法:

public static void main(String[] args) {
  try {
    FileInputStream inputStream = new FileInputStream(new File("test"));
    Throwable var2 = null;

    try {
      System.out.println(inputStream.read());
    } catch (Throwable var12) {
      var2 = var12;
      throw var12;
    } finally {
      if (inputStream != null) {
        if (var2 != null) {
          try {
            inputStream.close();
          } catch (Throwable var11) {
            var2.addSuppressed(var11);
          }
        } else {
          inputStream.close();
        }
      }

    }

  } catch (IOException var14) {
    throw new RuntimeException(var14.getMessage(), var14);
  }
}

3.3 異常抑制

通過反編譯的代碼,大家可能注意到代碼中有一處對(duì)異常的特殊處理:

var2.addSuppressed(var11);

這是try-with-resource語(yǔ)法涉及的另外一個(gè)知識(shí)點(diǎn),叫做異常抑制。當(dāng)對(duì)外部資源進(jìn)行處理(例如讀或?qū)懀r(shí),如果遭遇了異常,且在隨后的關(guān)閉外部資源過程中,又遭遇了異常,那么你catch到的將會(huì)是對(duì)外部資源進(jìn)行處理時(shí)遭遇的異常,關(guān)閉資源時(shí)遭遇的異常將被“抑制”但不是丟棄,通過異常的getSuppressed方法,可以提取出被抑制的異常。

四、總結(jié)

1、當(dāng)一個(gè)外部資源的句柄對(duì)象實(shí)現(xiàn)了AutoCloseable接口,JDK7中便可以利用try-with-resource語(yǔ)法更優(yōu)雅的關(guān)閉資源,消除板式代碼。

2、try-with-resource時(shí),如果對(duì)外部資源的處理和對(duì)外部資源的關(guān)閉均遭遇了異常,“關(guān)閉異常”將被抑制,“處理異常”將被拋出,但“關(guān)閉異常”并沒有丟失,而是存放在“處理異?!钡谋灰种频漠惓A斜碇?。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI