你真的知道嗎?catch、finally和return哪個先執行

来源:https://www.cnblogs.com/rupeng/archive/2023/08/02/17599580.html
-Advertisement-
Play Games

我的一位朋友前陣子遇到一個問題,問題的核心就是try……catch……finally中catch和finally代碼塊到底哪個先執。這個問題看起來很簡單,當然是“catch先執行、finally後執行”了?真的是這樣嗎? 有下麵一段C#代碼,請問這段代碼的執行結果是什麼? public static ...


我的一位朋友前陣子遇到一個問題,問題的核心就是try……catch……finally中catch和finally代碼塊到底哪個先執。這個問題看起來很簡單,當然是“catch先執行、finally後執行”了?真的是這樣嗎?

有下麵一段C#代碼,請問這段代碼的執行結果是什麼?

public static void Main(string[] args)

{

      try

      {

            A();

      }

      catch

      {

            Console.WriteLine("catch!!!");

      }

}

static void A()

{

      try

      {

            throw new Exception();

      }

      finally

      {

             Console.WriteLine("finally!!!");

      }

}

 

A()方法的try代碼塊中拋出了異常,而A方法沒有處理這個異常,所以Main方法的catch代碼塊會捕獲這個異常,但是A()方法中又有finally代碼塊,那麼到底是異常拋出後先執行Main方法的catch代碼塊呢還是先執行A()方法中的finally代碼塊呢?運行一下程式就能看出來,是finally代碼塊執行,結果如下所示。

finally!!!

catch!!!

 

為什麼呢?這需要從方法調用的異常對象如何傳遞給被調用方法講起。在一段代碼調用一個方法的時候,被調用的方法會把返回值、異常對象等放到一個特定的位置,這個位置叫做Stack Frame,調用者代碼會從這個特定的位置獲得被調用方法的返回值、異常對象等信息。因此,無論是throw異常的時候還是return返回值的時候,被調用的方法只是把異常對象或者返回值放到了這個特定的位置,在return或者throw執行之後,如果方法中還有finally等沒有執行完成的代碼,那麼這些代碼仍然會在return、throw之後繼續執行,然後方法執行才會結束,之後調用這個方法的代碼才會從Stack Frame中讀取到返回值或者獲取到被調用的方法拋出的異常對象。因此,上面的代碼才會先執行finally然後才執行catch。

明白了這個道理,請回答一下,下麵代碼的執行結果是什麼?

public static void Main(string[] args)

{

      try

{

A();

}

catch(Exception ex)

{

Console.WriteLine(ex.Message);

}

}

static void A()

{

      try

      {

            throw new Exception("aa");

      }

      finally

      {

            throw new Exception("bb");

      }

}

 

上面這是一段很特殊的代碼,在try代碼塊中拋出了一個異常(信息是aa),在finally中也拋出了一個異常(信息是bb),那麼程式實際列印出來的異常信息是什麼呢?上面程式執行結果是“bb”。通過上面的分析不難理解其原理:try代碼塊中的throw new Exception("aa")把方法的異常對象設置為Exception("aa"),而finall代碼塊中的throw new Exception("bb")又把方法的異常對象修改為Exception("bb"),因此最終方法拋出的異常對象是Exception("bb")。

接下來,我們再來捉弄一下方法的返回值,我們嘗試在finally代碼塊中修改方法的返回值。不幸的是(也可以說,幸運的是),C#禁止我們在finally代碼塊使用return語句,不過我們可以在Java中做這樣的嘗試,如下Java代碼所示:

public static void main(String[] args)

{

      System.out.println(A());

}

static int A()

{

      try

      {

            return 1;

      }

      finally

      {

            return 2;

      }

}

 

我們在try代碼塊中通過return 1把方法的返回值設置為1,但是在finally代碼塊中又把方法的返回值設置為2,因此方法的最終返回值就是2。

綜上所述,一個方法中通過return設定返回值或者throw拋出異常的時候,方法並沒有立即返回,只是在Stack Frame上保存了這個返回值或者異常對象,然後會繼續執行finally中的代碼,如果我們在finally代碼塊中修改了返回值或者拋出了新的異常,那麼最終的調用中獲得的返回值或者捕獲的對象就是修改後的返回值或者異常對象。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ## 一、問題是怎麼發現的 最近有個新系統開發完成後要上線,由於系統調用量很大,所以先對核心介面進行了一次壓力測試,由於核心介面中基本上只有純記憶體運算,所以預估核心介面的壓測QPS能夠達到上千。 壓測容器配置:4C8G 先從10個併發開始進行發壓,結果cpu一下就飆升到了100%,但是核心介面的qp ...
  • 編程基礎常識 一、註釋 1、對代碼的說明與解釋,它不會被編譯執行,也不會顯示在編譯結果中 2、註釋分為:單行註釋和多行註釋 3、用#號開始,例如:#這是我的第一個python程式 4、註釋可以寫在單獨一行,也可以寫在一句代碼後面 5、不想執行編譯,又不能刪除的代碼,可以先用#註釋掉,代碼批量註釋用C ...
  • 要解決多線程併發問題,常見的手段無非就幾種。加鎖,如使用synchronized,ReentrantLock,加鎖可以限制資源只能被一個線程訪問;CAS機制,如AtomicInterger,AtomicBoolean等原子類,通過自旋的方式來嘗試修改資源;還有本次我們要介紹的ThreadLocal類 ...
  • MQ(Message Queue)作為一種用於實現非同步通信的技術,具有重要的作用和應用場景。在面試過程中,MQ相關的問題經常被問到,因此瞭解MQ的用途和設計原則是必不可少的。本文總結了MQ的常見面試題,包括MQ的作用、產品選型、消息不丟失的保證、消息消費的冪等性、消息順序的保證、消息的高效讀寫、分佈... ...
  • 狀態機是有限狀態自動機的簡稱,是現實事物運行規則抽象而成的一個數學模型。狀態機,也就是 State Machine ,不是指一臺實際機器,而是指一個數學模型。說白了,一般就是指一張狀態轉換圖。 ...
  • EOF,為End Of File的縮寫,通常在文本的最後存在此字元表示資料結束。 在微軟的DOS和Windows中,讀取數據時終端不會產生EOF。此時,應用程式知道數據源是一個終端(或者其它“字元設備”),並將一個已知的保留的字元或序列解釋為文件結束的指明;最普遍地說,它是ASCII碼中的替換字元( ...
  • 本文介紹了在沒有 Spring Boot 和 Starter 之前,開發人員在使用傳統的 Spring XML 開發 Web 應用時需要引用許多依賴,並且需要大量編寫 XML 代碼來描述 Bean 以及它們之間的依賴關係。也瞭解瞭如何利用 SPI 載入自定義標簽來載入 Bean 併進行註入。 ...
  • ### 歡迎訪問我的GitHub > 這裡分類和彙總了欣宸的全部原創(含配套源碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### 本篇概覽 - 本文是《quarkus依賴註入》系列的第 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...