欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java.try catch finally 的執(zhí)行順序說明

 更新時間:2020年10月20日 09:53:40   作者:qq_45239139  
這篇文章主要介紹了Java.try catch finally 的執(zhí)行順序說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

示例1:

public static String hello() {
    String s = "商務(wù)";
    try {
      return s;
    } catch (Exception e) {
      return "catch進(jìn)來了";
    } finally {
      s = "你好世界";
      return s;
    }
  }

返回結(jié)果:你好世界,此時的返回順序是 finally > try

示例2:

public static String hello() {
    String s = "商務(wù)";
    try {
      return s;
    } catch (Exception e) {
      return "catch進(jìn)來了";
    } finally {
      s = "你好世界";
    }
  }

返回結(jié)果:商務(wù)

因為在 try 處會進(jìn)行 s 值的緩存

示例3:

public static void hehe() {
    int a = 10;
    try {
      a += 1;
      System.out.println(a);
      throw new Exception("catch測試");
    } catch (Exception e) {
      e.printStackTrace();
      a += 10;
      System.out.println(a);
    } finally {
      a += 10000;
      System.out.println(a);
    }
    System.out.println(a);
  }

返回結(jié)果:

11
21
10021
10021

故意在 try 處拋出一個異常進(jìn)入 catch,此時返回的順序是 try > catch > finally > 最后一行代碼

補充知識:Java異常獲取中try-catch-finally塊執(zhí)行順序

最近看面試題,發(fā)現(xiàn)這個比較好玩,try-catch-finally塊的執(zhí)行順序問題。

一般認(rèn)為,finally最后執(zhí)行,做收尾工作,無論try塊是否捕獲異常,最后finally都會工作。但是這樣還是比較籠統(tǒng),如果沒有catch,而是將異常拋出,讓其他方法處理,那么是先進(jìn)入其他方法還是先執(zhí)行finally?如果try塊中return了,那么finally還執(zhí)行不執(zhí)行?進(jìn)一步,如果try、finally全部有return,那么執(zhí)行是怎樣的過程?

確實,異常這些還是最早學(xué)Java的時候?qū)W的,當(dāng)時似乎也沒考慮這么多。借此機(jī)會研究一下異常獲取的順序。

節(jié)省時間,直接結(jié)論:

try->catch->finally按順序執(zhí)行,不管是否有異常,不管try中有什么操作,就算是return,也得往后稍稍,最后這個方法一定是要執(zhí)行finally。

如果try中拋出異常,而異常是留給上層方法處理,那么在拋出后,仍然運行finally,然后再回溯到上層。

自然,如果try中有return——也算是回溯了,返回值會存在棧中等待,等finally運行之后再回溯。

而如果finally中有return,那么直接從finally中結(jié)束方法。

如果在方法中直接結(jié)束程序,即調(diào)用System.exit()方法,那么就直接結(jié)束了,此時finally是不執(zhí)行的。由此可以認(rèn)為,特殊情況導(dǎo)致程序的退出是可能導(dǎo)致一些問題的。畢竟finally一般寫的是關(guān)閉對象、資源的代碼。

通過代碼分析:

先寫了一個包含情況比較多的例子:

package me.iwts; 
public class Main{
  public static int rank;
 
  public static void solve1() throws Exception{
    try {
      System.out.println("solve 1 try,rank: "+rank++);
      throw new Exception("throw by solve 1");
    }finally {
      System.out.println("solve 1 finally,rank: "+rank++);
    }
  }
 
  public static void solve2(){
    try{
      System.out.println("solve 2 try,rank: "+rank++);
      solve1();
    }catch (Exception ex){
      System.out.println("catch exception : "+ex.getMessage()+",rank: "+rank++);
    }finally {
      System.out.println("solve 1 finally,rank: "+rank++);
    }
  }
 
  public static void main(String args[]) {
    rank = 1;
    solve2();
    System.out.println("over");
  }
}

rank是計數(shù)。可以看到,整體上是先調(diào)用solve2方法,在其中調(diào)用solve1,solve1拋出了一個異常,讓solve2捕獲處理。大家可以先自己猜一下。

下面是返回答案:

solve 2 try,rank: 1
solve 1 try,rank: 2
solve 1 finally,rank: 3
catch exception : throw by solve 1,rank: 4
solve 1 finally,rank: 5
over

根據(jù)上面的結(jié)果可以分析:try-catch-finally執(zhí)行順序:首先是try執(zhí)行,如果發(fā)生異常,那么直接捕獲異常,最后執(zhí)行finally。但是,如果拋出異常,例如在solve1方法中,throw了一個異常,那么不會立刻回溯到上一個方法,而是仍然執(zhí)行finally。

通過solve1的執(zhí)行,我們可以認(rèn)為,finally之前的所有代碼,正常執(zhí)行,但是返回之類的,全部被“卡”了下來,只有在finally執(zhí)行之后,才能繼續(xù)執(zhí)行。

這里就又有疑惑了,一般認(rèn)為throw了一個異常,就算是回溯了,為什么finally仍然執(zhí)行了?如果這個不夠明顯,那么再看這個代碼:

package me.iwts; 
public class Main{
  public static int solve(){
    try{
      System.out.println("try");
      return 1;
    }finally {
      System.out.println("finally");
      return 2;
    }
  }
 
  public static void main(String args[]) {
    System.out.println(solve());
  }
}

返回值是多少?

try
finally
2

try塊都已經(jīng)return了,最后為什么是返回的return2?并且try塊確實是運行了。再改一下代碼:

package me.iwts; 
public class Main{
  public static int solve(){
    int i = 1;
    try{
      System.out.println("try");
      return i++;
    }finally {
      System.out.println("finally");
      return i;
    }
  }
 
  public static void main(String args[]) {
    System.out.println(solve());
  }
}

注意,try塊返回了i++,那么我們debug就能看出來return這句到底是執(zhí)行還是沒執(zhí)行,那么有這樣的圖:

可以看到,return確實是執(zhí)行的。

所以,認(rèn)為finally是無論怎樣一定在方法的最后結(jié)束前執(zhí)行的。搜了一些資料,是說finally會在方法結(jié)束之前執(zhí)行,而之前所有的執(zhí)行,包括return,全部都停留在棧中,而finally最終執(zhí)行后才繼續(xù)。所以這樣也能解釋,第一次代碼本應(yīng)該回溯的代碼執(zhí)行完finally后才回溯,return的時候也是等finally執(zhí)行之后再執(zhí)行。

或許“return的時候也是等finally執(zhí)行之后再執(zhí)行”這句話又引出了一個問題:finally究竟是直接運行完結(jié)束,還是運行完之后再回到原來return的地方?

這里我們可以把i++換成++i,結(jié)果就不截圖了——finally就是最終執(zhí)行,如果有return,直接從finally返回。

還有一種情況,直接結(jié)束程序會怎么樣?

package me.iwts; 
public class Main{
  public static void solve(){
    try{
      System.out.println("try");
      System.exit(0);
    }finally {
      System.out.println("finally");
    }
  }
 
  public static void main(String args[]) {
    solve();
  }
}

結(jié)果:

try

強制結(jié)束大過天。由此,也可以認(rèn)為特殊情況導(dǎo)致程序直接結(jié)束,不會執(zhí)行finally。因為finally一般寫的都是關(guān)閉對象、資源的代碼,所以這些特殊情況導(dǎo)致的程序強制結(jié)束,可能會引發(fā)一些問題的。

以上這篇Java.try catch finally 的執(zhí)行順序說明就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java基礎(chǔ)之二叉搜索樹的基本操作

    Java基礎(chǔ)之二叉搜索樹的基本操作

    發(fā)現(xiàn)許多小伙伴還不清楚Java二叉搜索樹的基本操作,今天特地整理了這篇文章,文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)Java的小伙伴很有幫助,需要的朋友可以參考下
    2021-05-05
  • spring boot Rabbit高級教程(最新推薦)

    spring boot Rabbit高級教程(最新推薦)

    RabbitMQ的消息過期是基于追溯方式來實現(xiàn)的,也就是說當(dāng)一個消息的TTL到期以后不一定會被移除或投遞到死信交換機(jī),而是在消息恰好處于隊首時才會被處理,本篇文章給大家介紹spring boot Rabbit高級教程,感興趣的朋友一起看看吧
    2023-10-10
  • Java 集合概覽(小結(jié))

    Java 集合概覽(小結(jié))

    這篇文章主要介紹了Java 集合概覽(小結(jié)),詳細(xì)的介紹了集合的概念和接口等。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-06-06
  • 利用Java中Calendar計算兩個日期之間的天數(shù)和周數(shù)

    利用Java中Calendar計算兩個日期之間的天數(shù)和周數(shù)

    Java 語言的Calendar(日歷),Date(日期),和DateFormat(日期格式)組成了Java標(biāo)準(zhǔn)的一個基本但是非常重要的部分。日期是商業(yè)邏輯計算一個關(guān)鍵的部分。下面這篇文章就給大家介紹了如何利用Java中Calendar計算兩個日期之間的天數(shù)和周數(shù),下面來一起看看吧。
    2016-12-12
  • 分析Java設(shè)計模式之組合模式

    分析Java設(shè)計模式之組合模式

    組合模式是一種對象的行為模式。將對象組合成樹形結(jié)構(gòu)以表示“部分-整體”的層次結(jié)構(gòu)。組合模式使得用戶對單個對象和組合對象的使用具有一致性。它的本質(zhì)是統(tǒng)一葉子對象和組合對象。它的目的是讓客戶端不再區(qū)分操作的是組合對象還是葉子對象,而是以一個統(tǒng)一的方式來操作
    2021-06-06
  • Maven 的配置文件路徑讀取方法

    Maven 的配置文件路徑讀取方法

    這篇文章主要介紹了Maven 的配置文件路徑讀取方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-06-06
  • Java怎么獲取當(dāng)前時間、計算程序運行時間源碼詳解(超詳細(xì)!)

    Java怎么獲取當(dāng)前時間、計算程序運行時間源碼詳解(超詳細(xì)!)

    有的時候,我們需要查看某一段代碼的性能如何,最為簡單的方式,可以通過計算該段代碼執(zhí)行的耗時,來進(jìn)行簡單的判斷,這篇文章主要給大家介紹了關(guān)于Java怎么獲取當(dāng)前時間、計算程序運行時間的相關(guān)資料,需要的朋友可以參考下
    2024-07-07
  • springboot中縮短一個url鏈接的實現(xiàn)

    springboot中縮短一個url鏈接的實現(xiàn)

    縮短 URL 是現(xiàn)代應(yīng)用程序中常見的需求,通常用于減少長 URL 的長度,使其更易于分享,URL 縮短服務(wù)的核心思路是將長 URL 映射到一個唯一的短代碼,本文主要介紹了springboot中縮短一個url鏈接的實現(xiàn),感興趣的可以了解一下
    2024-09-09
  • 關(guān)于IDEA無法預(yù)覽Markdown文件的解決思路

    關(guān)于IDEA無法預(yù)覽Markdown文件的解決思路

    在IntelliJ IDEA中,有時Markdown文件無法預(yù)覽可能是因為文件關(guān)聯(lián)設(shè)置不正確或配置信息錯誤,首先,檢查IDE的File Types設(shè)置,確保.md和.markdown后綴已正確注冊,其次,對照官方配置信息,調(diào)整Markdown設(shè)置
    2024-09-09
  • 集成apollo動態(tài)日志取締logback-spring.xml配置

    集成apollo動態(tài)日志取締logback-spring.xml配置

    這篇文章主要為大家介紹了集成apollo動態(tài)日志取締logback-spring.xml配置的過程內(nèi)容詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-02-02

最新評論