6

我在我的 Android 应用程序中使用 rxAndroid 和 rxKotlin 来异步处理网络请求。现在我想仅在单击 Snackbar 按钮后重试失败的网络请求。

我现在的代码:

val citiesService = ApiFactory.citiesService

citiesService.cities()
    .subscribeOn(Schedulers.newThread()) // fetch List<String>
    .flatMap { Observable.from(it) }     // convert to sequence of String
    .flatMap { city ->
        citiesService.coordinates(city)  // fetch DoubleArray
            .map { City(city, it) }      // convert to City(String, DoubleArray)
        }
    .toList()
    .observeOn(AndroidSchedulers.mainThread())
    .doOnNext {
        listView.setOnItemClickListener { adapterView, view, position, id ->
            onItemClick(it[position])
        }
    }
    .map { it.map { it.getName(activity) } }
    .subscribe(
        { listAdapter = setupAdapter(it) },
        { showErrorSnackbar() }  // handle error
    )

fun showErrorSnackbar() {
        Snackbar.make(listView, getString(R.string.not_available_msg), Snackbar.LENGTH_INDEFINITE)
                .setAction(getString(R.string.snack_retry_btn), {
                    // retry observable
                })
                .show()
    }

城市改造界面:

interface CitiesService {

    @GET("api/v1/cities")
    fun cities(): Observable<List<String>>

    @GET("api/v1/cities/{city}/coordinates")
    fun coordinates(@Path("city") city: String): Observable<DoubleArray>
}

API工厂:

object ApiFactory {

    val citiesService: CitiesService
        get() = retrofit.create(CitiesService::class.java)

    private val retrofit: Retrofit
        get() = Retrofit
            .Builder()
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .addConverterFactory(GsonConverterFactory.create())
            .baseUrl(BASE_URL)
            .build()
}

我怎样才能以这种方式重新启动 observable?

4

1 回答 1

1

我可以建议你真正的反应方式而不是命令方式。

在 subscribe() 方法之后插入此代码:

.retryWhen(retryHandler -> 
           retryHandler.flatMap(nothing -> retrySubject.asObservable()))
.subscribe()

更新主题只是:

@NonNull
private final PublishSubject<Void> retrySubject = PublishSubject.create();

在快餐栏上单击调用此方法:

public void update() {
    retrySubject.onNext(null);
}

retryWhen 方法之上的所有内容都将被重做。

尽管使用这种方法,错误永远不会传递给订阅者,但您可以将错误调节添加到 retryHandler 平面地图,但这是另一回事。

PS 抱歉,这是带有 retrolambdas 的 Java 代码,但您可以轻松地将其转换为 Kotlin。

于 2017-03-28T21:47:05.057 回答