试着在4天内解决这个问题,请帮助。我正在创建一个带有rest API的应用程序( retrofit ),尝试从Google samples https://github.com/googlesamples/android-architecture-components/tree/master/GithubBrowserSample中实现LiveDataCallAdapter,但是rest没有调用适配器方法adapt来从服务器获取响应。我只编辑了NetworkBoundResourse (在没有数据库的情况下工作)试图放置断点,在我启动repo (登录)后,LiveDataCallAdapter fun adapt ( call.enequeue不想启动)调试不调用
以下是我的代码片段thx
提供我的服务实例
@Singleton
@Provides
fun provideRetrofit(): BrizSmartTVService {
return Retrofit.Builder()
.baseUrl(baseUrl)
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(LiveDataCallAdapterFactory())
.build()
.create(BrizSmartTVService::class.java)
}这是我的LiveDataCallAdapterFactory和LiveDataCallAdapter
class LiveDataCallAdapterFactory : Factory() {
override fun get(
returnType: Type,
annotations: Array<Annotation>,
retrofit: Retrofit
): CallAdapter<*, *>? {
if (getRawType(returnType) != LiveData::class.java) {
return null
}
val observableType = getParameterUpperBound(0, returnType as ParameterizedType)
val rawObservableType = getRawType(observableType)
if (rawObservableType != ApiResponse::class.java) {
throw IllegalArgumentException("type must be a resource")
}
if (observableType !is ParameterizedType) {
throw IllegalArgumentException("resource must be parameterized")
}
val bodyType = getParameterUpperBound(0, observableType)
return LiveDataCallAdapter<Any>(bodyType)
}
}class LiveDataCallAdapter<R>(private val responseType: Type) :
CallAdapter<R, LiveData<ApiResponse<R>>> {
override fun responseType() = responseType
override fun adapt(call: Call<R>): LiveData<ApiResponse<R>> {
return object : LiveData<ApiResponse<R>>() {
private var started = AtomicBoolean(false)
override fun onActive() {
super.onActive()
if (started.compareAndSet(false, true)) {
Log.d("TAG", ": onActive Started ");
call.enqueue(object : Callback<R> {
override fun onResponse(call: Call<R>, response: Response<R>) {
Log.d("TAG", ": $response");
postValue(ApiResponse.create(response))
}
override fun onFailure(call: Call<R>, throwable: Throwable) {
Log.d("TAG", ": ${throwable.localizedMessage}");
postValue(ApiResponse.create(throwable))
}
})
}
}
}
}
}这是我的NetworkBoundResourse (仅适用于网络)
abstract class NetworkBoundResource<RequestType> {
private val result = MediatorLiveData<Resource<RequestType>>()
init {
setValue(Resource.loading(null))
fetchFromNetwork()
}
@MainThread
private fun setValue(newValue: Resource<RequestType>) {
if (result.value != newValue) {
result.value = newValue
}
}
private fun fetchFromNetwork() {
val apiResponse = createCall()
result.addSource(apiResponse) { response ->
result.removeSource(apiResponse)
when (response) {
is ApiSuccessResponse -> {
setValue(Resource.success(processResponse(response)))
}
is ApiErrorResponse -> {
onFetchFailed()
setValue(Resource.error(response.errorMessage, null))
}
}
}
}
protected fun onFetchFailed() {
}
fun asLiveData() = result as LiveData<Resource<RequestType>>
@WorkerThread
protected open fun processResponse(response: ApiSuccessResponse<RequestType>) = response.body
@MainThread
protected abstract fun createCall(): LiveData<ApiResponse<RequestType>>
}我的Repo类
@Singleton
class AuthApiRepo @Inject constructor(
val apiService: BrizSmartTVService
) {
fun authLoginPass(login: String, password: String): LiveData<Resource<AuthResponse>> {
return object : NetworkBoundResource<AuthResponse>() {
override fun createCall(): LiveData<ApiResponse<AuthResponse>> {
val authLogPassBody = AuthLogPassBody(login,password,"password")
Log.d("TAG", ": $authLogPassBody");
return apiService.authLoginPass(authLogPassBody)
}
}.asLiveData()
}
}和我的AuthResponse类
class AuthResponse {
@SerializedName("token_type")
var tokenType: String? = null
@SerializedName("access_token")
var accessToken: String? = null
@SerializedName("refresh_token")
var refreshToken: String? = null
@SerializedName("user_id")
var userId: String? = null
@SerializedName("expires_in")
var expiresIn: Long = 0
@SerializedName("portal_url")
var portalUrl: String? = null
}我的ViewModel类,我从这里开始调用
class AuthViewModel @Inject constructor(private val authApiRepo: AuthApiRepo) : ViewModel() {
private var _isSigned = MutableLiveData<Boolean>()
val isSigned: LiveData<Boolean>
get() = _isSigned
fun signIn(login: String, password: String) {
authApiRepo.authLoginPass(login, password)
val authRespons = authApiRepo.authLoginPass(login, password)
Log.d("TAG", ": " + authRespons.value.toString());
//here will by always data null and status LOADING
}
override fun onCleared() {
super.onCleared()
}
}发布于 2019-04-21 02:29:05
所以伙计们,我终于找到了一个解决方案。对于MVVM (实时数据)的经验丰富的人来说,这很简单,但我是MVVM的初学者,当我开始学习MVVM时,我的大脑爆炸了。所以,问题是我从ViewModel订阅了Repo livedata,而不是从视图(在我的例子中是Fragment)。在我锁定了实时数据观察者链之后,视图- ViewModel -回购-服务-一切正常。Thx通向所有
https://stackoverflow.com/questions/55764251
复制相似问题