kotlin協程異常處理之-try catch

来源:https://www.cnblogs.com/tangZH/archive/2023/08/21/17300624.html
-Advertisement-
Play Games

- [kotlin協程小記](http://77blogs.com/?p=73 "kotlin協程小記") - [協程的async使用](http://77blogs.com/?p=77 "協程的async使用") - [kotlin協程異常處理之-try catch ](http://77blog ...


一、try catch

try catch是否一定有效呢?未必,來看一下:

1、withContext

import kotlinx.coroutines.*

fun main() = runBlocking {
    launch {
        println("launch start")
        try {
            withContext(Dispatchers.IO) {
                // 可能拋出異常
            }
        } catch (ex: Exception) {
            println("withContext caught: ${ex.message}")
        }
        println("launch end")
    }
}

withContext是一個掛起函數,它會暫停當前協程的執行,等待傳遞進來的協程上下文切換後繼續執行。當在withContext內部發生異常時,異常會被傳遞迴到withContext函數的調用者,也就是當前協程的上一級代碼中,進而可以被try-catch塊捕獲到。

2、launch

import kotlinx.coroutines.*

fun main() = runBlocking {
    try {
        launch {
            println("launch start")
            // 可能拋出異常
            println("launch end")
        }
    } catch (ex: Exception) {
        println("launch caught: ${ex.message}")
    }
}

try {
    GlobalScope.launch {
        throw NullPointerException()
    }
} catch (e :Exception) {
    e.printStackTrace()
}

launch啟動的協程是獨立於調用它的協程之外的一個新的協程,它沒有直接的上級協程來捕獲它的異常,因此try-catch在協程外部捕獲不到協程中的異常。
事實證明,只要是launch的協程,無論是子協程還是根協程,都無法被捕獲。比如:

GlobalScope.launch {
    try {
        launch {
            Log.d("MainActivity_", "launch-> threadName->" + Thread.currentThread().name)
            throw NullPointerException()
        }
    } catch (e :Exception) {
        e.printStackTrace()
    }
}

一樣會崩潰。

如果將try catch放於內部:

import kotlinx.coroutines.*

fun main() = runBlocking {
     launch {
         try {
            // 可能拋出異常
         } catch (ex: Exception) {
            println("launch caught: ${ex.message}")
         }
         println("launch end")
	 }
}

這樣便可以捕獲得到異常了。

3、async

(1)內部async

GlobalScope.launch {
    try {
        val deferredResult: Deferred<Int> = async {
            Log.d("AsyncTest", "throw before")
            throw Exception("async function exception")
            Log.d("AsyncTest", "throw after")
        }
        deferredResult.await()
    } catch (ex: Exception) {
        Log.d("AsyncTest", "${ex.message}")
    }
}

輸出:

D/AsyncTest: throw before
D/AsyncTest: async function exception

但是程式奔潰了,可以捕獲異常,但是會崩。


(2)、將try catch放於內部:

GlobalScope.launch {
    val deferredResult: Deferred<Int> = async {
        try {
            Log.d("AsyncTest", "throw before")
            throw Exception("async function exception")
            Log.d("AsyncTest", "throw after")
        } catch (e: java.lang.Exception) {
            Log.d("AsyncTest", "${e.message}")
        }
    }
    deferredResult.await()
}

輸出:

D/AsyncTest: throw before
D/AsyncTest: async function exception

可以捕獲異常,並且程式不會崩潰。


(3)、使用GlobalScope.async

GlobalScope.launch {
    try {
        val deferredResult: Deferred<Int> = GlobalScope.async {
            Log.d("AsyncTest", "throw before")
            throw Exception("async function exception")
            Log.d("AsyncTest", "throw after")
        }
        deferredResult.await()
    } catch (ex: Exception) {
        Log.d("AsyncTest", "${ex.message}")
    }
}

輸出:

D/AsyncTest: throw before
D/AsyncTest: async function exception

可以捕獲異常,並且程式不會崩潰。


(4)、只對deferredResult.await()try catch

GlobalScope.launch {
    val deferredResult: Deferred<Int> = GlobalScope.async {
        Log.d("AsyncTest", "throw before")
        throw Exception("async function exception")
        Log.d("AsyncTest", "throw after")
    }
    try {
        deferredResult.await()
    } catch (e: Exception) {
        Log.d("AsyncTest", "${e.message}")
    }
}

輸出:

D/AsyncTest: throw before
D/AsyncTest: async function exception

可以捕獲異常,並且程式不會崩潰。

結論

1、withContext是一個掛起函數,它會暫停當前協程的執行,等待傳遞進來的協程上下文切換後繼續執行。當在withContext內部發生異常時,異常會被傳遞迴到withContext函數的調用者,也就是當前協程的上一級代碼中,進而可以被try-catch塊捕獲到。
2、launch啟動的協程是獨立於調用它的協程之外的一個新的協程,它沒有直接的上級協程來捕獲它的異常,因此try-catch在協程外部捕獲不到協程中的異常。
3、async如果啟動的是子協程,那麼代碼執行到 throw 異常的時候就拋出了異常,與是否調用await方法無關,這個異常可以用try-catch捕獲但是會引起崩潰。
4、async開啟一個根協程,在調用await方法時候會拋出異常,這個異常可以用try-catch捕獲不引起崩潰。


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

-Advertisement-
Play Games
更多相關文章
  • 近日全球領先的IT市場研究和咨詢公司IDC正式發佈《中國關係型資料庫軟體市場跟蹤報告-數據倉庫市場Add-on》報告華為雲GaussDB(DWS)憑藉領先的技術和優異的市場表現榮獲“雙第一”。 ...
  • 最近因需求改動新增了一些資料庫表,但是在定義表結構時,具體列屬性的選擇有些不知其所以然,索引的添加也有遺漏和不規範的地方,所以我打算為創建一個高性能表的過程以實戰的形式寫一個專題,以此來學習和鞏固這些知識。 ...
  • NineData與實時數據倉庫廠商SelectDB完成產品相容互認證,實現了軟體相互相容、功能完善、運行穩定且性能優異。雙方將持續助力數據管理與大數據分析業務的融合,幫助企業實現數字化轉型,提高效率、降低成本,並滿足客戶業務需求。SelectDB成立於2022年初,團隊來自知名互聯網和雲計算公司,核... ...
  • # 前言 本篇來介紹一下redis pipeline,主要是由於最近一次在幫開發同學review代碼的時候,發現對redis有個迴圈操作可以優化。場景大概是這樣的,根據某個uid要從redis查詢一批數據,每次大概1000個key左右,如果查得到就返回,否則查db,然後寫回緩存。由於每次要查的key ...
  • 鎖共有多種演算法,在併發場景中都是被常常用到,想必大家都已爐火純青般.....巴特!我們還有後浪同學們可能不熟悉,那我在這裡聊下鎖的用法和使用場景。 ...
  • - 1.說在前面的話 - 2.安裝employees測試庫 - 3.觀測SQL運行狀態 - - 3.1 觀測SQL運行時的記憶體消耗 - 3.2 觀測SQL運行時的其他開銷 - 3.3 觀測SQL運行進度 > 感知SQL運行時的狀態 ## 1. 說在前面的話 在MySQL里,一條SQL運行時產生多少磁 ...
  • 簡述 上篇文章介紹瞭如何搭建 prometheus 監控體系,監控 linux 伺服器,這篇文章跟大家介紹如何監控 redis,以及我們要關註的指標都有哪些 監控 redis 需要關註什麼指標 在《聊聊監控》這篇文章,介紹了 google 提出的監控四個黃金指標(沒看過的朋友可以看看這篇文章),下麵 ...
  • 轉載請標明出處:https://www.cnblogs.com/tangZH/p/17307406.html - [kotlin協程小記](http://77blogs.com/?p=73 "kotlin協程小記") - [協程的async使用](http://77blogs.com/?p=77 " ...
一周排行
    -Advertisement-
    Play Games
  • WPF本身不支持直接的3D繪圖,但是它提供了一些用於實現3D效果的高級技術。 如果你想要在WPF中進行3D繪圖,你可以使用兩種主要的方法: WPF 3D:這是一種在WPF應用程式中創建3D圖形的方式。WPF 3D提供了一些基本的3D形狀(如立方體、球體和錐體)以及一些用於控制3D場景和對象的工具(如 ...
  • 一、XML概述 XML(可擴展標記語言)是一種用於描述數據的標記語言,旨在提供一種通用的方式來傳輸和存儲數據,特別是Web應用程式中經常使用的數據。XML並不預定義標記。因此,XML更加靈活,並且可以適用於廣泛的應用領域。 XML文檔由元素(element)、屬性(attribute)和內容(con ...
  • 從今年(2023)三月份開始,Github開始強制用戶開啟兩步驗證2FA(雙因數)登錄驗證,毫無疑問,是出於安全層面的考慮,畢竟Github賬號一旦被盜,所有代碼倉庫都會毀於一旦,關於雙因數登錄的必要性請參見:別讓你的伺服器(vps)淪為肉雞(ssh暴力破解),密鑰驗證、雙向因數登錄值得擁有。 雙因 ...
  • 第一題 下列代碼輸入什麼? public class Test { public static Test t1 = new Test(); { System.out.println("blockA"); } static { System.out.println("blockB"); } publi ...
  • 本文主要涉及的問題:用ElementTree和XPath讀寫XML文件;解決ElementTree新增元素後再寫入格式不統一的問題;QTableWidget單元格設置控制項 ...
  • QStandardItemModel 類作為標準模型,主打“類型通用”,前一篇水文中,老周還沒提到樹形結構的列表,本篇咱們就好好探討一下這貨。 還是老辦法,咱們先做示例,然後再聊知識點。下麵這個例子,使用 QTreeView 組件來顯示數據,使用的列表模型比較簡單,只有一列。 #include <Q ...
  • 一、直充內充(充值方式) 直充: 包裝套餐直接充值到上游API系統。【PID/Smart】 (如:支付寶、微信 話費/流量/語音/簡訊 等 充值系統)。 內充(套餐打包常見物聯卡系統功能): 套餐包裝 適用於不同類型套餐 如 流量、簡訊、語音 等。 (目前已完善流量邏輯) 二、套餐與計費產品 計費產 ...
  • 在前面幾天中,我們學習了Dart基礎語法、可迭代集合,它們是Flutter應用研發的基本功。今天,我們繼續學習Flutter應用另一個必須掌握知識點:非同步編程(即Future和async/await)。它類似於Java中的FutureTask、JavaScript中的Promise。它是後續Flut... ...
  • 針對改動範圍大、影響面廣的需求,我通常會問上線了最壞情況是什麼?應急預案是什麼?你帶開關了嗎?。當然開關也是有成本的,接下來本篇跟大家一起交流下高頻發佈支撐下的功能開關技術理論與實踐結合的點點滴滴。 ...
  • 1.d3.shuffle D3.shuffle() 方法用於將數組中的元素隨機排序。它使用 Fisher–Yates 洗牌演算法,該演算法是無偏的,具有最佳的漸近性能(線性時間和常數記憶體)。 D3.shuffle() 方法的語法如下: d3.shuffle(array, [start, end]) 其中 ...