Использовать методы дооснащения более выразительным способом
Я хочу сделать void enqueue(Callback<T> callback);
блок кода вызова метода более выразителен, вот что у меня обычно
request.enqueue(object : Callback<MyModel> {
override fun onFailure(call: Call<MyModel>?, t: Throwable?) {
//
}
override fun onResponse(call: Call<MyModel>?, response: Response<MyModel>?) {
//
}
})
И то, что я хочу и имею в виду, - чтобы этот код блокировался более чистым способом , удалял те ключевые слова override, object, Callback и делал что-то подобное:
request.enqueue({throwable, response → })
Я думаю, что это можно как-то улучшить, используя расширения и функции высшего порядка. Кто-нибудь знает, как это можно сделать?
Ответы
Ответ 1
Вот как я делаю это с помощью функции расширения и класса
fun<T> Call<T>.enqueue(callback: CallBackKt<T>.() -> Unit) {
val callBackKt = CallBackKt<T>()
callback.invoke(callBackKt)
this.enqueue(callBackKt)
}
class CallBackKt<T>: Callback<T> {
var onResponse: ((Response<T>) -> Unit)? = null
var onFailure: ((t: Throwable?) -> Unit)? = null
override fun onFailure(call: Call<T>, t: Throwable) {
onFailure?.invoke(t)
}
override fun onResponse(call: Call<T>, response: Response<T>) {
onResponse?.invoke(response)
}
}
тогда вы можете использовать это так
request.enqueue {
onResponse = {
// do
}
onFailure = {
// do
}
}
Ответ 2
Учитывая следующую функцию:
fun <T> callback(fn: (Throwable?, Response<T>?) -> Unit): Callback<T> {
return object : Callback<T> {
override fun onResponse(call: Call<T>, response: retrofit2.Response<T>) = fn(null, response)
override fun onFailure(call: Call<T>, t: Throwable) = fn(t, null)
}
}
Вы можете использовать это с помощью Retrofit следующим образом:
request.enqueue(callback({ throwable, response ->
response?.let { callBack.onResponse(response.body() ?: RegisterResponse()) }
throwable?.let { callBack.onFailed(throwable.message!!) })
В качестве альтернативы вы можете определить эту другую версию обратного вызова:
fun <T> callback2(success: ((Response<T>) -> Unit)?, failure: ((t: Throwable) -> Unit)? = null): Callback<T> {
return object : Callback<T> {
override fun onResponse(call: Call<T>, response: retrofit2.Response<T>) { success?.invoke(response) }
override fun onFailure(call: Call<T>, t: Throwable) { failure?.invoke(t) }
}
}
который можно использовать следующим образом:
request.enqueue(callback2(
{ r -> callBack.onResponse(r.body()) },
{ t -> callBack.onFailed(t.message) }))
Ответ 3
Что вы можете сделать, так это (это Java, поскольку я не знаю много Kotlin, но это должно быть очень похоже):
public class CallbackWrapper<T> implements Callback<T> {
private Wrapper<T> wrapper;
public CallbackWrapper(Wrapper<T> wrapper) {
this.wrapper = wrapper;
}
public void onFailure(Call<T> call, Throwable t) {
wrapper.onResult(t, null);
}
public void onResponse(Call<T> call, Response<T> response) {
wrapper.onResult(null, response);
}
public static interface Wrapper<T> {
void onResult(Throwable t, Response<T> response);
}
}
Который вы можете использовать как:
call.enqueue(new CallbackWrapper((throwable, reponse) - > {...}));
обновление решения для kotlin:
Исходя из этого, CallBackWrapper
выглядит следующим образом:
typealias wrapper<T> = (t: Throwable?, response: Response<T>?) -> Unit
class CallbackWrapper<T>(val wrapper: wrapper<T>) : Callback<T> {
override fun onFailure(call: Call<T>?, t: Throwable?) = wrapper.invoke(t,null)
override fun onResponse(call: Call<T>?, response: Response<T>?) = wrapper.invoke(null, response)
}
и использовать его так же, как Java.
Ответ 4
Я использовал функцию расширения в Call
чтобы написать выразительный и общий метод enqueue
.
fun<T> Call<T>.onEnqueue(actOnSuccess: (Response<T>) -> Unit, actOnFailure: (t: Throwable?) -> Unit) {
this.enqueue(object: Callback<T> {
override fun onFailure(call: Call<T>?, t: Throwable?) {
actOnFailure(t)
}
override fun onResponse(call: Call<T>?, response: Response<T>) {
actOnSuccess(response)
}
})
}
Это можно затем использовать как:
request.onEnqueue {
actOnSuccess = {
doOnSuccess()
}
actOnFailure = {
doOnFailure()
}
}
В блоках actOnSuccess и actOnFailure кода, it
будет относиться к Response
и Throwable
объектам соответственно, и может быть использован соответствующим образом. Например, внутри блока кода actOnFailure -
actOnFailure = {
doOnFailure()
//it.message //'it' refers to the Throwable object.
}
Ответ 5
Вы можете создать функцию расширения, как это
inline fun <T> Call<T>.addEnqueue(
crossinline onSuccess: (response: Response<T>) -> Unit = { response: Response<T> -> },
crossinline onFail: (t: Throwable) -> Unit = { throwable: Throwable ->}
):Callback<T> {
val callback = object : Callback<T> {
override fun onFailure(call: Call<T>, t: Throwable) {
onFail(t)
}
override fun onResponse(call: Call<T>, response: Response<T>) {
onSuccess(response)
}
}
enqueue(callback)
return callback
}
и использовать это в своей деятельности или фрагменте
service?.fetchUser()?.addEnqueue(
onSuccess = {
doOnSuccess(it)
},
onFail = {
doOnFail(it)
}
)