前言:
在android開發(fā)中开皿,關(guān)于架構(gòu)的設(shè)計忱屑,不管是MVC,還是MVP,抑或是MVVM壳坪,http的網(wǎng)絡(luò)請求封裝(這里是指二次封裝舶得,基于 OkHttp Retrofit 的再封裝),前篇文章說的不夠仔細(xì)爽蝴。
github也有很多比較優(yōu)秀的二次封裝沐批,本人也收益匪淺纫骑,這里我也給大家分享一下我的經(jīng)驗
力求簡潔!>藕ⅰ先馆!優(yōu)雅!L杀颉煤墙!實用!O苡怠仿野!
前置:大家對協(xié)程 以及 flow 有一定的了解,這里就不說線程與協(xié)程的區(qū)別了
只需要記住kotlin協(xié)程僅僅是線程的框架江解,僅僅是方便開發(fā)者進(jìn)行異步操作的框架设预,并沒有任何性能優(yōu)勢
ok,開擼
1: 請求是個消耗資源的操作犁河,所以必須在子線程中進(jìn)行鳖枕,request方法就屬于IO線程
2: 響應(yīng)的處理是在主線程進(jìn)行,協(xié)程已經(jīng)自動切回主線程桨螺,resp方法就是在主線程中
android Http獲取網(wǎng)絡(luò)數(shù)據(jù)的過程就是 請求(IO) + 響應(yīng)(UI) + 線程切換的過程宾符。
沒有那么多彎彎繞繞,秉著這個原則就很好理解與運用了灭翔,有了kotlin 協(xié)程就更加簡潔了魏烫。
Retrofit + 協(xié)程, 一行代碼就實現(xiàn) HTTP請求的封裝 簡潔的令人發(fā)指
fun <T> launchData(
scope: CoroutineScope = GlobalScope,
request: suspend CoroutineScope.() -> Res<T>,
resp:(T?)->Unit
) {
scope.launch {
try {
resp( request() .data)
} catch (t: Throwable) {
catchThr(t) // 異常處理
}
}
}
那具體如何使用呢 三鞭肯定能倒
第一鞭: 使用retrofit UserApi (名字隨意 )定義好網(wǎng)絡(luò)接口
intferface UserApi {
/**
* 接化發(fā)
*/
@GET("article/jhf")
suspend fun jhf():Res<String?>
}
第二鞭: 注入 userApi(單例或者注解都可以)
提示!!!:如果在Activity 發(fā)起http請求 scope 建議使用傳入 lifecycleScope,
如果在viewmodel 發(fā)起請求建議使用viewModelScope ,筆者以后者為例
/**
* 接化發(fā)
*/
class viewmodel : BaseViewModel{
...
fun jhf(ok: (String?) -> Unit) {
// 請求 響應(yīng)
launchData( { userApi.jhf() }, { ok(it) })
...或者...
launchData(
requset = { userService.jhf() },
resp = { ok(it) }
)
}
}
這里userApi.jhf() 就是flow 中 request()方法肝箱。ok()方法接收的值就是flow中的resp()方法返回的值哄褒,即網(wǎng)絡(luò)數(shù)據(jù)返回的結(jié)果值。
從數(shù)據(jù)請求到結(jié)果返回煌张,業(yè)務(wù)層僅需一行代碼呐赡。
第三鞭:在activity或者fragment直接調(diào)用viewmodel中方法,結(jié)果直接返回骏融,
class MainActivity : BaseActivity(){
override fun onCreate(savedInstanceState: Bundle?) {
...
//這里的it 就是網(wǎng)絡(luò)請求返回的結(jié)果
viewModel.jhf {textView.text = it}
}
}
三鞭打完!!! 如果 發(fā)生網(wǎng)絡(luò)異常該如何處理呢链嘀?
兩種處理方式
1 默認(rèn)彈toast 原先邏輯不變
fun <T> launchData(
scope: CoroutineScope = GlobalScope,
request: suspend CoroutineScope.() -> Res<T>,
resp:(T?)->Unit,
error:(String)->Unit = {toast(it)} //toat(“網(wǎng)絡(luò)異常”)
) {
scope.launch {
try {
val baseData = request()
if(baseData.code == RESP_OK){
resp( baseData.data)
}else{
error(baseData.msg)
}
} catch (t: Throwable) {
catchThr(t) // 異常處理
}
}
}
2 不彈toast 自行處理
在viewmodel 調(diào)用的時候這樣寫
/**
* 接化發(fā)
*/
fun jhf(ok:(String?)->Unit,error: ()->Unit){
launchData( { userService.jhf() }, { ok(it) }, error = { error() })
...或者...
launchData(
request = { userService.jhf() },
resp = { ok(it) },
error = { error() }
)
}
在activity或者frament調(diào)用的這樣寫
viewModel.jhf(
ok = { textView.text = it},
error = { } // 這里處理異常
)
或者這樣都可以
viewModel.jhf({ textView.text = it}){
//這里處理異常
}
關(guān)于協(xié)程的取消
//當(dāng)組件結(jié)束時档玻,會取消協(xié)程內(nèi)的任務(wù)
override fun onCleared() {
viewModelScope.cancel()
}
后記
上面的封裝剔除了一些狀態(tài)的判斷!!!
關(guān)于showLoading ,hideLoading ,以及頁面狀態(tài)的轉(zhuǎn)換都在Base中做了統(tǒng)一處理
具體的代碼怀泊,筆者已經(jīng)上傳github
https://github.com/ruirui1128/jetpack-hilt-flow-mvvm.git