首頁 > Java > java教程 > 使用Java finally語句的一些注意事項

使用Java finally語句的一些注意事項

零下一度
發布: 2017-06-25 10:44:15
原創
1615 人瀏覽過

網路上有很多人探討Java中異常捕獲機制try...catch...finally區塊中的finally語句是不是一定會被執行?很多人都說不是,當然他們的答案是正確的,經過我試驗,至少有兩種情況下finally語句是不會被執行的:

(1)try語句沒有被執行到,如在try語句之前就回傳了,這樣finally語句就不會執行,這也說明了finally語句被執行的必要而非充分條件是:對應的try語句一定被執行到。

(2)在try區塊中有System.exit(0);這樣的語句,System.exit(0);是終止Java虛擬機器JVM的,連JVM都停止了,所有都結束了,當然finally語句也不會被執行到。

 

當然還有很多人探討Finally語句的執行與return的關係,頗為讓人迷惑,不知道finally語句是在try的return之前執行還是之後執行?我也是一頭霧水,我覺得他們的說法都不正確,我覺得應該是:finally語句是在try的return語句執行之後,return返回之前執行。這樣的說法有點矛盾,也許是我表述不太清楚,以下我給出自己試驗的一些結果和範例進行佐證,有什麼問題歡迎大家提出來。

1. finally語句在return語句執行之後return回傳之前執行的。

使用Java finally語句的一些注意事項
public class FinallyTest1 {

    public static void main(String[] args) {
        
        System.out.println(test1());
    }

    public static int test1() {
        int b = 20;

        try {
            System.out.println("try block");

            return b += 80; 
        }
        catch (Exception e) {

            System.out.println("catch block");
        }
        finally {
            
            System.out.println("finally block");
            
            if (b > 25) {
                System.out.println("b>25, b = " + b);
            }
        }
        
        return b;
    }
    
}
登入後複製
使用Java finally語句的一些注意事項

執行結果是:

try block
finally block
b>25, b = 100
100
登入後複製
登入後複製

說明return語句已經執行了再去執行finally語句,不過並沒有直接返回,而是等finally語句執行完了再返回結果。

如果覺得這個例子還不足以說明這個情況的話,下面再加個例子加強證明結論:

使用Java finally語句的一些注意事項
public class FinallyTest1 {

    public static void main(String[] args) {
        
        System.out.println(test11());
    }
    
    public static String test11() {
        try {
            System.out.println("try block");

           return test12();
      } finally {
           System.out.println("finally block");
       }
  }

  public static String test12() {
       System.out.println("return statement");

       return "after return";
   }
    
}
登入後複製
使用Java finally語句的一些注意事項

運行結果為:

try block
return statement
finally block
after return
登入後複製

說明try中的return語句先執行了但並沒有立即返回,等到finally執行結束後再

這裡大家可能會想:如果finally裡也有return語句,那麼是不是就直接返回了,try中的return就不能返回了?看下面。

2. finally區塊中的return語句會覆寫try區塊中的return回傳。

使用Java finally語句的一些注意事項
public class FinallyTest2 {

    public static void main(String[] args) {

        System.out.println(test2());
    }

    public static int test2() {
        int b = 20;

        try {
            System.out.println("try block");

            return b += 80;
        } catch (Exception e) {

            System.out.println("catch block");
        } finally {

            System.out.println("finally block");

            if (b > 25) {
                System.out.println("b>25, b = " + b);
            }

            return 200;
        }

        // return b;
    }

}
登入後複製
使用Java finally語句的一些注意事項

執行結果是:

try block
finally block
b>25, b = 100
200
登入後複製

這表示finally裡的return直接回傳了,就不管try中是否還有回傳語句,這裡還有個小細節要注意,finally裡加上return過後,finally外面的return b就變成不可到達語句了,也就是永遠不能被執行到,所以需要註解掉否則編譯器報錯。

這裡大家可能又想:如果finally裡沒有return語句,但修改了b的值,那麼try中return回傳的是修改後的值還是原值?看下面。

3. 如果finally語句中沒有return語句覆寫回傳值,那麼原來的回傳值可能會因為finally裡的修改而改變也可能不變。

測試案例1:

使用Java finally語句的一些注意事項
#
public class FinallyTest3 {

    public static void main(String[] args) {

        System.out.println(test3());
    }

    public static int test3() {
        int b = 20;

        try {
            System.out.println("try block");

            return b += 80;
        } catch (Exception e) {

            System.out.println("catch block");
        } finally {

            System.out.println("finally block");

            if (b > 25) {
                System.out.println("b>25, b = " + b);
            }

            b = 150;
        }

        return 2000;
    }

}
登入後複製
使用Java finally語句的一些注意事項

執行結果是:

try block
finally block
b>25, b = 100
100
登入後複製
登入後複製

 

 

測試案例2:

 

使用Java finally語句的一些注意事項
#
import java.util.*;

public class FinallyTest6
{
    public static void main(String[] args) {
        System.out.println(getMap().get("KEY").toString());
    }
     
    public static Map<string> getMap() {
        Map<string> map = new HashMap<string>();
        map.put("KEY", "INIT");
         
        try {
            map.put("KEY", "TRY");
            return map;
        }
        catch (Exception e) {
            map.put("KEY", "CATCH");
        }
        finally {
            map.put("KEY", "FINALLY");
            map = null;
        }
         
        return map;
    }
}</string></string></string>
登入後複製
使用Java finally語句的一些注意事項
#

运行结果是:

FINALLY
登入後複製

为什么测试用例1中finally里的b = 150;并没有起到作用而测试用例2中finally的map.put("KEY", "FINALLY");起了作用而map = null;却没起作用呢?这就是Java到底是传值还是传址的问题了,具体请看精选30道Java笔试题解答,里面有详细的解答,简单来说就是:Java中只有传值没有传址,这也是为什么map = null这句不起作用。这同时也说明了返回语句是try中的return语句而不是 finally外面的return b;这句,不相信的话可以试下,将return b;改为return 294,对原来的结果没有一点影响。

这里大家可能又要想:是不是每次返回的一定是try中的return语句呢?那么finally外的return b不是一点作用没吗?请看下面。

4. try块里的return语句在异常的情况下不会被执行,这样具体返回哪个看情况。

使用Java finally語句的一些注意事項
public class FinallyTest4 {

    public static void main(String[] args) {

        System.out.println(test4());
    }

    public static int test4() {
        int b = 20;

        try {
            System.out.println("try block");

            b = b / 0;

            return b += 80;
        } catch (Exception e) {

            b += 15;
            System.out.println("catch block");
        } finally {

            System.out.println("finally block");

            if (b > 25) {
                System.out.println("b>25, b = " + b);
            }

            b += 50;
        }

        return 204;
    }

}
登入後複製
使用Java finally語句的一些注意事項

运行结果是:

try block
catch block
finally block
b>25, b = 35
85
登入後複製
这里因 为在return之前发生了除0异常,所以try中的return不会被执行到,而是接着执行捕获异常的catch 语句和最终的finally语句,此时两者对b的修改都影响了最终的返回值,这时return b;就起到作用了。当然如果你这里将return b改为return 300什么的,最后返回的就是300,这毋庸置疑。

这里大家可能又有疑问:如果catch中有return语句呢?当然只有在异常的情况下才有可能会执行,那么是在finally之前就返回吗?看下面。

5. 当发生异常后,catch中的return执行情况与未发生异常时try中return的执行情况完全一样。

使用Java finally語句的一些注意事項
public class FinallyTest5 {

    public static void main(String[] args) {

        System.out.println(test5());
    }

    public static int test5() {
        int b = 20;

        try {
            System.out.println("try block");
            
            b = b /0;

            return b += 80;
        } catch (Exception e) {

            System.out.println("catch block");
            return b += 15;
        } finally {

            System.out.println("finally block");

            if (b > 25) {
                System.out.println("b>25, b = " + b);
            }

            b += 50;
        }

        //return b;
    }

}
登入後複製
使用Java finally語句的一些注意事項

运行结果如下:

try block
catch block
finally block
b>25, b = 35
35
登入後複製

说明了发生异常后,catch中的return语句先执行,确定了返回值后再去执行finally块,执行完了catch再返回,finally里对b的改变对返回值无影响,原因同前面一样,也就是说情况与try中的return语句执行完全一样。

 

最后总结:finally块的语句在try或catch中的return语句执行之后返回之前执行且finally里的修改语句可能影响也可能不影响try或catch中 return已经确定的返回值,若finally里也有return语句则覆盖try或catch中的return语句直接返回。

学习Java的同学注意了!!!
学习过程中遇到什么问题或者想获取学习资源的话,欢迎加入Java学习交流群:159610322   我们一起学Java!

以上是使用Java finally語句的一些注意事項的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板