PageKeyedDataSource load连续调用后

Eli*_*ber 5 kotlin android-architecture-components android-paging

我有一个连续调用loadAfter的PageKeyedDataSource,并且所有项目都多次添加到Recyclerview中。从API方面来看,null lastEvaluatedKey表示给我第一页,这对于为什么它不断调用以获取第一页还是A是有意义的。如果没有更多数据要获取(又名params.key == null?和B ,则它不应该停止) t是COMPARATOR在适配器不允许添加相同的项目多次吗?我在想什么?

PageKeyedDataSource.kt

class ReservationsPageKeyedDataSource(private val retryExecutor: Executor) : PageKeyedDataSource<String, Reservation?>() {

    private var retry: (() -> Any)? = null

    val initialLoad = MutableLiveData<PagingNetworkState>()

    fun retryAllFailed() {
        val prevRetry = retry
        retry = null
        prevRetry?.let {
            retryExecutor.execute {
                it.invoke()
            }
        }
    }

    override fun loadInitial(
        params: LoadInitialParams<String>,
        callback: LoadInitialCallback<String, Reservation?>
    ) {
        val request = Api.reservationsService.getReservations(dateType = RERVATIONS_DATE_TYPE.future, last = null)

        initialLoad.postValue(PagingNetworkState.LOADING)

        // triggered by a refresh, execute in sync
        try {
            val response = request.execute()
            val originalData = response.body()?.result?.reservations
            val data = mutableListOf<Reservation>()
            // some data munipulation
            retry = null
            initialLoad.postValue(PagingNetworkState.LOADED)

            callback.onResult(
                data.toList(),
                null,
                response.body()?.result?.lastEvaluatedKey.toString()
            )
        } catch (ioException: IOException) {
            retry = {
                loadInitial(params, callback)
            }
            val error = PagingNetworkState.error(ioException.message ?: "unknown error")
            initialLoad.postValue(error)
        }
    }

    override fun loadBefore(
        params: LoadParams<String>,
        callback: LoadCallback<String, Reservation?>
    ) {
        // no-op
    }

    override fun loadAfter(
        params: LoadParams<String>,
        callback: LoadCallback<String, Reservation?>
    ) {

        // I tried adding an if statement here to check if the params.key is null or not but that didn't help

        Api.reservationsService.getReservations(dateType = RERVATIONS_DATE_TYPE.future, last = params.key)
            .enqueue(object : Callback<ReservationListResponse> {
                override fun onFailure(call: Call<ReservationListResponse>, t: Throwable) {
                    retry = { loadAfter(params, callback) }
                }

                override fun onResponse(
                    call: Call<ReservationListResponse>,
                    response: Response<ReservationListResponse>
                ) {
                    if (response.isSuccessful) {
                        val data = response.body()?.result?.reservations
                        retry = null
                        callback.onResult(
                            data.orEmpty(),
                            response.body()?.result?.lastEvaluatedKey.toString()
                        )
                    } else {
                        retry = { loadAfter(params, callback) }
                    }
                }
            })
    }
}
Run Code Online (Sandbox Code Playgroud)

PagedListAdapter中的比较器:

companion object {
        val COMPARATOR = object : DiffUtil.ItemCallback<Reservation>() {
            override fun areContentsTheSame(oldItem: Reservation, newItem: Reservation): Boolean =
                oldItem == newItem

            override fun areItemsTheSame(oldItem: Reservation, newItem: Reservation): Boolean =
                oldItem.id == newItem.id
        }
    }
Run Code Online (Sandbox Code Playgroud)

小智 5

另一个可能的原因是,如果您在 ScrollView 内部使用 Recycleview,则 loadAfter 将被无限调用,直到数据完成。RecycleView会自动处理滚动,所以不要在scrollView中使用它。


Chr*_*aik 1

代码看起来大部分都很好,但奇怪的是,您通过 toString 将响应的下一个标记部分转换为字符串,并将其用作密钥。

而不是PageKeyedDataSource<String, Reservation?>尝试PageKeyedDataSource<KeyType, Reservation>(从上面的代码中不确定该键的类型是什么)。

然后,您可以直接从 API 获取下一个令牌,并将其传递到lastAPI 的参数中,而无需修改它。

您还应该使用不可为空的Reservation- 分页库期望加载的项目为非空,因为空值被保留用于表示占位符:https ://developer.android.com/reference/androidx/paging/DataSource#implementing-a -数据源

  • 当我有一项也 loadAfter 调用并重复时,为什么? (2认同)