詳解RxJava2 Retrofit2 網(wǎng)絡(luò)框架簡潔輕便封裝
前言
RxJava2、Retrofit2火了有一段時(shí)間了,前段時(shí)間給公司的項(xiàng)目引入了這方面相關(guān)的技術(shù),在此記錄一下相關(guān)封裝的思路。
需求
封裝之前要先明白需要滿足哪些需求。
- RxJava2銜接Retrofit2
- Retrofit2網(wǎng)絡(luò)框架異常的統(tǒng)一處理
- 兼容fastjson(可選)
- RxJava2內(nèi)存泄漏的處理
- 異步請(qǐng)求加入Loading Dialog
依賴
implementation 'io.reactivex.rxjava2:rxandroid:2.0.1' implementation 'io.reactivex.rxjava2:rxjava:2.1.3' implementation 'com.squareup.retrofit2:retrofit:2.3.0' implementation 'com.squareup.retrofit2:adapter-rxjava2:2.3.0' implementation 'com.squareup.okhttp3:okhttp:3.9.0' implementation 'com.trello.rxlifecycle2:rxlifecycle-components:2.2.0' implementation 'com.alibaba:fastjson:1.1.59.android'//可選其它框架比如Gson
RxJava2銜接Retrofit2
先封裝一個(gè)網(wǎng)絡(luò)框架的管理類,方便調(diào)用
public class RxHttp {
private final String BASE_URL = "https://github.com/";
private Map<String, Retrofit> mRetrofitMap = new HashMap<>();
private RxHttp() {
}
/**
* 單例模式
* @return
*/
public static RxHttp getInstance() {
return RxHttpHolder.sInstance;
}
private static class RxHttpHolder{
private final static RxHttp sInstance = new RxHttp();
}
public Retrofit getRetrofit(String serverUrl) {
Retrofit retrofit;
if (mRetrofitMap.containsKey(serverUrl)) {
retrofit = mRetrofitMap.get(serverUrl);
} else {
retrofit = createRetrofit(serverUrl);
mRetrofitMap.put(serverUrl, retrofit);
}
return retrofit;
}
public SyncServerService getSyncServer(){
return getRetrofit(BASE_URL).create(SyncServerService.class);
}
/**
*
* @param baseUrl baseUrl要以/作為結(jié)尾 eg:https://github.com/
* @return
*/
private Retrofit createRetrofit(String baseUrl) {
OkHttpClient client = new OkHttpClient().newBuilder()
.readTimeout(30, TimeUnit.SECONDS)
.connectTimeout(30, TimeUnit.SECONDS)
.retryOnConnectionFailure(true)
.build();
return new Retrofit.Builder()
.baseUrl(baseUrl)
.addConverterFactory(FastJsonConverterFactory.create())
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.client(client)
.build();
}
}
Restful風(fēng)格接口
public interface SyncServerService {
@GET("service/mobile/IsLatestVersion.ashx")
Observable<Response<String>> getLatestVersion(@Query("SoftwareID") String SoftwareID,
@Query("ClientVersion") String ClientVersion);
}
服務(wù)端返回的基本類型,在導(dǎo)入類的時(shí)候特別需要注意區(qū)分該Response類型
public class Response<T> {
public int ret;//約定 -1為server返回?cái)?shù)據(jù)異常 200為正常范圍
public String msg;
public T data;
public int getRet() {
return ret;
}
public void setRet(int ret) {
this.ret = ret;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public T getData() {
return data;
}
public void setData(T data) {
this.data = data;
}
}
fastjson的支持
由于項(xiàng)目中采用了fastjson,square尚未實(shí)現(xiàn)對(duì)fastjson的支持,但是保留了代碼的擴(kuò)展,這邊可以自己封裝一下fastjson的轉(zhuǎn)換器。
public class FastJsonConverterFactory extends Converter.Factory {
private final SerializeConfig mSerializeConfig;
private FastJsonConverterFactory(SerializeConfig serializeConfig) {
this.mSerializeConfig = serializeConfig;
}
public static FastJsonConverterFactory create() {
return create(SerializeConfig.getGlobalInstance());
}
public static FastJsonConverterFactory create(SerializeConfig serializeConfig) {
return new FastJsonConverterFactory(serializeConfig);
}
@Override
public Converter<?, RequestBody> requestBodyConverter(Type type, Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
return new FastJsonRequestBodyConverter<>(mSerializeConfig);
}
@Override
public Converter<ResponseBody, ?> responseBodyConverter(Type type, Annotation[] annotations, Retrofit retrofit) {
return new FastJsonResponseBodyConvert<>(type);
}
}
final class FastJsonRequestBodyConverter<T> implements Converter<T, RequestBody> {
private final MediaType MEDIA_TYPE = MediaType.parse("application/json; charset=UTF-8");
private SerializeConfig mSerializeConfig;
public FastJsonRequestBodyConverter(SerializeConfig serializeConfig) {
this.mSerializeConfig = serializeConfig;
}
@Override
public RequestBody convert(T value) throws IOException {
return RequestBody.create(MEDIA_TYPE, JSON.toJSONBytes(value, mSerializeConfig));
}
}
final class FastJsonResponseBodyConvert<T> implements Converter<ResponseBody, T> {
private Type mType;
public FastJsonResponseBodyConvert(Type type) {
this.mType = type;
}
@Override
public T convert(ResponseBody value) throws IOException {
return JSON.parseObject(value.string(), mType);
}
}
數(shù)據(jù)返回統(tǒng)一處理
public abstract class BaseObserver<T> implements Observer<Response<T>> {
@Override
public final void onNext(@NonNull Response<T> result) {
if (result.getRet() == -1) {
onFailure(new Exception(result.getMsg()), result.getMsg());//該異??梢詤R報(bào)服務(wù)端
} else {
onSuccess(result.getData());
}
}
@Override
public void onError(@NonNull Throwable e) {
onFailure(e, RxExceptionUtil.exceptionHandler(e));
}
@Override
public void onComplete() {
}
@Override
public void onSubscribe(@NonNull Disposable d) {
}
public abstract void onSuccess(T result);
public abstract void onFailure(Throwable e, String errorMsg);
}
下面加入了異常處理類
public class RxExceptionUtil {
public static String exceptionHandler(Throwable e){
String errorMsg = "未知錯(cuò)誤";
if (e instanceof UnknownHostException) {
errorMsg = "網(wǎng)絡(luò)不可用";
} else if (e instanceof SocketTimeoutException) {
errorMsg = "請(qǐng)求網(wǎng)絡(luò)超時(shí)";
} else if (e instanceof HttpException) {
HttpException httpException = (HttpException) e;
errorMsg = convertStatusCode(httpException);
} else if (e instanceof ParseException || e instanceof JSONException
|| e instanceof com.alibaba.fastjson.JSONException) {
errorMsg = "數(shù)據(jù)解析錯(cuò)誤";
}
return errorMsg;
}
private static String convertStatusCode(HttpException httpException) {
String msg;
if (httpException.code() >= 500 && httpException.code() < 600) {
msg = "服務(wù)器處理請(qǐng)求出錯(cuò)";
} else if (httpException.code() >= 400 && httpException.code() < 500) {
msg = "服務(wù)器無法處理請(qǐng)求";
} else if (httpException.code() >= 300 && httpException.code() < 400) {
msg = "請(qǐng)求被重定向到其他頁面";
} else {
msg = httpException.message();
}
return msg;
}
}
異步請(qǐng)求加入Loading Dialog
這個(gè)時(shí)候我們可以根據(jù)自己項(xiàng)目中統(tǒng)一封裝的dialog自行擴(kuò)展BaseObserver
public abstract class ProgressObserver<T> extends BaseObserver<T>{
private MaterialDialog mMaterialDialog;
private Context mContext;
private String mLoadingText;
public ProgressObserver(Context context){
this(context, null);
}
public ProgressObserver(Context context, String loadingText){
mContext = context;
mLoadingText = loadingText;
}
@Override
public void onSubscribe(@NonNull Disposable d) {
if (!d.isDisposed()) {
mMaterialDialog = new MaterialDialog.Builder(mContext).content(mLoadingText == null ? "正在加載中..."
: mLoadingText).isProgress(true).build();
mMaterialDialog.show();
}
}
@Override
public void onComplete() {
if (mMaterialDialog != null) {
mMaterialDialog.dismiss();
}
}
@Override
public void onError(@NonNull Throwable e) {
super.onError(e);
if (mMaterialDialog != null) {
mMaterialDialog.dismiss();
}
}
}
加入調(diào)度類,方便調(diào)用線程切換和解決內(nèi)存泄漏的問題
public class RxSchedulers {
public static <T> ObservableTransformer<T, T> observableIO2Main(final Context context) {
return upstream -> {
Observable<T> observable = upstream.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread());
return composeContext(context, observable);
};
}
public static <T> ObservableTransformer<T, T> observableIO2Main(final RxFragment fragment) {
return upstream -> upstream.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread()).compose(fragment.<T>bindToLifecycle());
}
private static <T> ObservableSource<T> composeContext(Context context, Observable<T> observable) {
if(context instanceof RxActivity) {
return observable.compose(((RxActivity) context).bindUntilEvent(ActivityEvent.DESTROY));
} else if(context instanceof RxFragmentActivity){
return observable.compose(((RxFragmentActivity) context).bindUntilEvent(ActivityEvent.DESTROY));
}else if(context instanceof RxAppCompatActivity){
return observable.compose(((RxAppCompatActivity) context).bindUntilEvent(ActivityEvent.DESTROY));
}else {
return observable;
}
}
}
講了那么多,那么如何使用這個(gè)封裝呢?下面來看下如何使用。
RxHttp.getInstance().getSyncServer().getLatestVersion("1", "1.0.0")
.compose(RxSchedulers.observableIO2Main(this))
.subscribe(new ProgressObserver<String>(this) {
@Override
public void onSuccess(String result) {
Toast.makeText(MainActivity.this, result, Toast.LENGTH_SHORT).show();
}
@Override
public void onFailure(Throwable e, String errorMsg) {
}
});
是不是封裝后的代碼顯得更為簡潔一點(diǎn)呢?以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- RxJava2.x實(shí)現(xiàn)定時(shí)器的實(shí)例代碼
- RxJava2.x+ReTrofit2.x多線程下載文件的示例代碼
- Android 使用 RxJava2 實(shí)現(xiàn)倒計(jì)時(shí)功能的示例代碼
- 基于Retrofit2+RxJava2實(shí)現(xiàn)Android App自動(dòng)更新
- RxJava2配置及使用詳解
- Android 用RxBinding與RxJava2實(shí)現(xiàn)短信驗(yàn)證碼倒計(jì)時(shí)功能
- RxJava2和Retrofit2封裝教程(整潔、簡單、實(shí)用)
- Rxjava2_Flowable_Sqlite_Android數(shù)據(jù)庫訪問實(shí)例
- 基于RxJava2實(shí)現(xiàn)的簡單圖片爬蟲的方法
- 談?wù)凴xJava2中的異常及處理方法
相關(guān)文章
使用Win10+Android+夜神安卓模擬器,搭建ReactNative開發(fā)環(huán)境
今天小編就為大家分享一篇關(guān)于使用Win10+Android+夜神安卓模擬器,搭建ReactNative開發(fā)環(huán)境,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2018-10-10
Android波紋擴(kuò)散效果之仿支付寶咻一咻功能實(shí)現(xiàn)波紋擴(kuò)散特效
這篇文章主要介紹了Android波紋擴(kuò)散效果之仿支付寶咻一咻功能實(shí)現(xiàn)波紋擴(kuò)散特效的相關(guān)資料,需要的朋友可以參考下2016-02-02
Android studio配置lambda表達(dá)式教程
Java 8的一個(gè)大亮點(diǎn)是引入Lambda表達(dá)式,使用它設(shè)計(jì)的代碼會(huì)更加簡潔。接下來通過本文給大家介紹Android studio配置lambda表達(dá)式教程,需要的朋友參考下吧2017-05-05
Android監(jiān)控和阻斷InputDispatching ANR的方法
如何在Java層實(shí)現(xiàn)異步監(jiān)控和阻斷InputDispatching ANR?我相信這是很多開發(fā)者都想要的功能,本篇,我們會(huì)通過“探索”兩種方案來實(shí)現(xiàn)在Java層監(jiān)控&阻斷的方法,需要的朋友可以參考下2024-04-04
Android編程實(shí)現(xiàn)ListView內(nèi)容無限循環(huán)顯示的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)ListView內(nèi)容無限循環(huán)顯示的方法,通過繼承Adapter類實(shí)現(xiàn)ListView中的數(shù)據(jù)無限循環(huán)顯示功能,需要的朋友可以參考下2017-06-06
Android 判斷所有字段是否已經(jīng)輸入的實(shí)例
今天小編就為大家分享一篇Android 判斷所有字段是否已經(jīng)輸入的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-03-03

