Resource的release 是在使用resource物件時,所特別需要注意的事,小則AP掛掉,大則機器損毀;所以凡是開發人員,在使用resource物件時,都要特別小心,像是Connection、File、I/O Stream等。可是根據莫菲定律,越擔心會發生的事就越會發生,code寫久了,總有一天你會忘記release resource…

適合閱讀
本篇適合以下人員閱讀:
- 知道Java不只是咖啡的人
- 會寫Hello Java的人
好吧,如果你不會寫程式,也可以看看,提供一些意見;如果你是dot Net高手,也請不吝指教。
正文
開發人員在開發的時候,除了bug之外,最怕的就是忘記關閉一些已使用資源物件。 1.7 版本以後,所有的Closeable物件都繼承了AutoCloseable;也同時提供了一個新的語法:try-with-resoure statement。 今天主要介紹這個try-with-resource的用法。
我們一般的try-catch寫法是這樣的:
public void normalTry() throws Exception {
    String sql = "This is a sql statement with parameter";
    Connection conn = null;
    PreparedStatement ps = null;
    ResultSet rs = null;
    try {
        conn = getConnection();// Any methodology to get connection.
        ps = conn.prepareStatement(sql);
        ps.setString(1, "This is parameter");
        rs = ps.executeQuery();
        while (rs.next()) {
            rs.getString(1);
        }
    } finally {
        try {
            rs.close();
            ps.close();
            conn.close();
        } catch (Exception e) {
            throw e;
        }
    }
}一般的try-catch-finally寫法,物件的宣告必需在try-catch之前,而且必需要有個finally來releaseCloseable的物件。1.7 以後你可以在try後面增加( )並在裡面放置一些需要close的物件。 這裡是一個try-with-resource的範例:
public void tryWithResource() throws Exception {
    String sql = "This is a sql statement";
    try (Connection conn = getConnection(); 
        PreparedStatement ps = conn.prepareStatement(sql)) {
        ps.setString(1, "This is parameter");
        try (ResultSet rs = ps.executeQuery(sql);) {
            while (rs.next()) {
                rs.getString(1);
            }
        }
    }
}使用新的try-with-resource寫法,程式行數由23行減為10行。
Try-with-resource也可以像一般try-catch的方式,搭配catch與finally服用。 簡單來說try-with-resource statement,就是這樣。
不過事情並不是這麼單純。
進階
我們把進階分成優點跟缺點來看…
優點
- Try-with-resource statement 可以防止開發人員因為忘記關閉resource導致系統 crash。 - 還記得我們一開始提到 - AutoCloseable嗎? 在try 裡面的東西不是阿貓阿狗都可以放進來,他必需要是- AutoCloseable的物件。原本1.6以前的- Closeable物件(其實是Interface),現在已經直接繼承- AutoCloseable,所以所有的- Closeable物件都是- AutoCloseable物件。 JVM會在離開try之後依建構的反向順序 呼叫- AutoCloseable.close()。- PS: - Closeable、- AutoCloseable都是interface。
- 可以減少程式碼。 不言自明,由範例來看,程式碼很明顯的減少了。 
缺點
- 程式碼可能不容易閱讀。 - tryWithResource的範例中,- ResultSet必需在- PreapreStatement設定參數後才能執行,才能建構,所以沒辦法放與- Connection、- PrepareStatement放在同一個- try()括號中,因此出現在巢狀的try敘述。 也許看久了會習慣,不過以現時的我來說,還沒那麼習慣。當然,這純屬個人偏好;但想像一下,這種情形如果重覆很多層,有時候,真的會很難搞清楚最後一個大括號,到底是對應那一個。
- 當其他錯誤與 - close()同時發生錯誤的話,- close()的錯誤會被蓋掉。- 考慮以下程式碼: - public void suppressException() throws Exception { String sql = "This is a sql statement"; try (Connection conn = getConnection(); PreparedStatement ps = conn.prepareStatement(sql)) { if (allwaysReturnTrue()) { throw new OperationNotSupportedException(); } }- 假設當 - Connection在- close()時發生錯誤,log只會看到- OperationNotSupportedException()所拋出的錯誤,不會看到- close()的錯誤。不過這有方法可以處理,就是使用- Throwable.getSuppressed()。- public void catchSuppressException() throws Exception { String sql = "This is a sql statement"; try (Connection conn = getConnection(); PreparedStatement ps = conn.prepareStatement(sql)) { if (allwaysReturnTrue()) { throw new OperationNotSupportedException(); } } } public static void main(String[] args) { TryWithRessourceDemo d=new TryWithRessourceDemo(); try { d.catchSuppressException(); } catch (Throwable e) { System.out.println("main"); System.out.println(e); Throwable[] suppressed = e.getSuppressed();//Use getSuppressed() to find suppressed exception for (int i = 0; i < suppressed.length; i++) { Throwable throwable = suppressed[i]; System.out.println(throwable); } } }
小結
本篇介紹了try-with-resource的用法與特性,好用,但用得太過很容造成閱讀障礙。 雖然close()不太容易會發生錯誤,但要注意suppressed exception的問題
同場加映
一開始學Java時,只道try-catch必需成雙成對,最多是只有finally可以選擇性的加與不加; 有人知道catch與finally都可以選擇加與不加嗎?
//我以為try-catch只有2種寫法
try{...} catch(){...}
try{...} catch(){...} finally{...}
//偶然看到高手這麼用
try{...} finally{...} 
 
沒有留言 :
張貼留言