内容简介:When designing how to handlenetwork calls in your app, you might think that you are done as soon as you define your interface. You are halfway there: you have defined the "happy" path. But what happens when not everything goes according to plan? That's the
When designing how to handlenetwork calls in your app, you might think that you are done as soon as you define your interface. You are halfway there: you have defined the "happy" path. But what happens when not everything goes according to plan? That's the other half.
As always, a reminder that there are no completely right or wrong answers here. It's rather a matter of choosing the most suitable approach depending on your requirements and your app architecture to make your life easier in the long run.
The straightforward way: try-catch statements
An error is by definition an exception to the normal flow of the app. Kotlin supports exception throwing and catching quite smoothly. So, all you have to do is throw the correct exception when an issue arises from the network library.
To make your code network library independent, a good approach is to create your own BaseNetworkException
class and create an exception class for each type of error that can happen.
viewModelScope.launch {
try {
callService()
}
catch (e: NoNetworkException) {
// Handle it
}
catch (e: AuthenticationNetworkException) {
// Handle it
}
}
The downside here is that you need to wrap every network call with a try-catch
statement. If you forgot about it and an error occurs, your app will crash.
The modern way: return Result
A network call can either succeed or fail. So a Result
class should represent exactly this: a success state with the data, or a failed state with the exception.
Kotlin has a built-in
Result
but when trying to use it as a return type you will get kotlin.Result' cannot be used as a return type
( why
). You can either create your own
Result
class
, use an open-source alternative
, or use a
hack to use the standard library Result
(not really recommended since it can break your build in the future).
viewModelScope.launch {
callService().fold(
{ data ->
// Successful call
},
{ error ->
if (error is NoNetworkException) {
// Hanlde it
}
else if (error is AuthenticationNetworkException) {
// Hanlde it
}
}
}
Using this approach you are forced to handle the potential error. The downside might be that it gets repetitive, especially if handling the same generic errors (but this can be fixed by using common handler methods).
The coroutine way: CoroutineExceptionHandler
The launch
coroutine builder accepts a default exception handler
parameter. The handler is activated only within the scope of the coroutine. This could be convenient for handling all the "common" exceptions (e.g. no internet connection). With this approach, there's no need to repeat how to handle common errors.
Any errors not handled by the coroutine exception handler will still have to be handled by one of the previous ways (or create specific CoroutineExceptionHandler
instances).
val genericErrorHandler = CoroutineExceptionHandler { _, error ->
if (error is NoNetworkException) {
// Hanlde it
}
else if (error is AuthenticationNetworkException) {
// Hanlde it
}
}
[...]
viewModelScope.launch(genericErrorHandler) {
callService()
}
Hopefully, by now you have a bit clearer picture on how to handle your network errors. Happy coding!
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
计算机组成(第 6 版)
Andrew S. Tanenbaum、Todd Austin / 刘卫东、宋佳兴 / 机械工业出版社 / 2014-8-19 / CNY 99.00
本书采用结构化方法来介绍计算机系统,书的内容完全建立在“计算机是由层次结构组成的,每层完成规定的功能”这一概念之上。作者对本版进行了彻底的更新,以反映当今最重要的计算机技术以及计算机组成和体系结构方面的最新进展。书中详细讨论了数字逻辑层、微体系结构层、指令系统层、操作系统层和汇编语言层,并涵盖了并行体系结构的内容,而且每一章结尾都配有丰富的习题。本书适合作为计算机专业本科生计算机组成与结构课程的教......一起来看看 《计算机组成(第 6 版)》 这本书的介绍吧!