Use Kotlin to simply encapsulate most common Retrofit+RxJava requests
Tip: If you don’t know much about Kotlin, RxLifecycle, Retrofit, RxJava, etc., check out the basics online
1. The convention is to add dependencies first
//Retrofit
implementation 'com. Squareup. Retrofit2: retrofit: 2.3.0'
implementation 'com. Squareup. Retrofit2: adapter - rxjava2:2.3.0'
//RxJava
implementation 'the IO. Reactivex. Rxjava2: rxandroid: 2.0.1'
implementation 'the IO. Reactivex. Rxjava2: rxjava: 2.0.1'
//rxlifecycle
implementation 'com. Trello. Rxlifecycle2: rxlifecycle - components: 2.1.0'// implement'com. Alibaba: fastjson: 1.2.34'
//kotlin
implementation "org.jetbrains.kotlin:kotlin-stdlib-jre7:$kotlin_version"//Kotlin reflection library Kotlin uses reflection to add implementation"org.jetbrains.kotlin:kotlin-reflect:$kotlin_version"
Copy the code
2. Encapsulate the request Retrofit interface
Typically we might have only one BaseUrl per application, where only the Retrofit interface is singleton encapsulated
public class RetrofitUtils {
private RetrofitUtils() { } private static class Tool { static IApiServer defaultServer = initRetrofitServer(NetConfig.BASE_URL); static IApiServer initRetrofitServer(String baseUrl) { int DEFAULT_TIME = 15; // Default timeout okHttpClient.builder Builder = new okhttpClient.builder (); builder.readTimeout(DEFAULT_TIME, TimeUnit.SECONDS); builder.connectTimeout(DEFAULT_TIME, TimeUnit.SECONDS); // builder.addInterceptor(new LoggingInterceptor()); // Use custom Log interceptor OkHttpClient client = builder.build();return new Retrofit.Builder()
.baseUrl(baseUrl)
.addConverterFactory(FastJsonConverterFactory.create())
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.client(client)
.build()
.create(IApiServer.class);
}
}
public static IApiServer getDefaultServer() {
returnTool.defaultServer; }}Copy the code
3. Request response encapsulation
Encapsulate returns that have a regular format, such as the following
{
"code": "200"."data": {}
"msg": "Success"
}
Copy the code
Each request returns a fixed code and MSG, and data is the real data. Define an entity class of the following format:
public class BaseResponse<T> {
@JSONField(name = "status")
public int status;
@JSONField(name = "data")
public T data;
@JSONField(name = "msg")
public String msg;
}
Copy the code
4. Request function encapsulation
Tips in ApiCodeType that encapsulates some of the state and errors can be written to suit your needs
For state interception, if the business is very complex, you can also extract into interceptors and other ways
Object RetrofitUtilsKt {/** * an Activity binding declares a unified function that periodically processes data requests * @Param Observable Request interface Observable instance * @param LifecycleProvider is empty if it is empty and does not bind declaration cycle * @param showDialog whether to display LoadView default display * @param onNext success callback * @param onError failure callback * @param showDialog loadView */ runRxA(Observable: Observable<T>, lifecycleProvider: LifecycleProvider<ActivityEvent>? , onNext: (data: T) -> Unit, onError: (codeType: ApiCodeType) -> Unit, showDialog: Boolean =true): Disposable { var dialog: AlertDialog? Lifecycle Val realObservable = nullif(lifecycleProvider ! = null) observable.compose(lifecycleProvider.bindUntilEvent(ActivityEvent.DESTROY))else
observable
returnRealObservable. SubscribeOn (Schedulers. IO ()). ObserveOn (AndroidSchedulers. MainThread ()). DoOnSubscribe {/ / request before Operations such as loadView are displayedif(lifecycleProvider is Activity && showDialog) { dialog = DefaultUtils.showProgressDialog(lifecycleProvider) } } .subscribe(Consumer<T> { dialog? .apply {dismiss()} // Data errorif (it == null || (it is BaseResponse<*> && it.data == null)) {
onError(ApiCodeType.DATA_ERROR)
return@consumer} // Intercepts part of the state based on business requirementsif(it is BaseResponse<*>) {when (it. Status) {apicodetype.logon_expired. code ->{// Intercept code}else -> onNext(it)
}
} else{ onNext(it) } }, Consumer<Throwable> { dialog? .apply { dismiss() } onError(ApiCodeType.OTHER_ERROR) }) } }Copy the code
Only functions that bind the Activity life cycle are shown here. If you need to bind the Fragment life cycle, change the generic name of the LifecycleProvider to FragmentEvent.
5. Extend the encapsulation of functions
For ordinary simple requests, we usually only care about three elements: the request interface (including parameters), the association lifecycle (activities, etc.), and the result callback.
In Kotlin, you can add extension functions for any type. With this feature, we write the following functions for one of the necessary types (the request interface) to make it easier to use:
fun <T> Observable<T>.runRxA(lifecycleProvider: LifecycleProvider<ActivityEvent>?,
onNext: (data: T) -> Unit,
onError: (codeType: ApiCodeType) -> Unit,
showDialog: Boolean = true) =
RetrofitUtilsKt.runRxA(this, lifecycleProvider, onNext, onError, showDialog)
Copy the code
6. Use
The single parameter lamdba in Kotlin does not need to be declared and can be expressed as IT. The code is commented as the current IT type.
RetrofitUtils .getDefaultServer() .getContractInformationEdit(bean) .runRxA(this, {// successful //it:BaseResponse<ContractInformationBean>}, {// failed //it:ApiCodeType toast(it.msg)})Copy the code
At the end
Hope this article is helpful to each partner, if wrong or doubt welcome to point out the message. 🙂