首先我们先不说封装思路,先说说这套框架都都具有哪些功能及如何使用;
功能
1 使用RxCache缓存机制,可自定义缓存过期时间,及数据分页缓存等功能。
2 统一的请求错误处理;
3 统一的网络状态判断处理;
4 基于HttpLoggingInterceptor的请求日志打印。
以上就是这套框架可以实现的功能,框架中并没有像其他的一样封装了ProgressBar
,因为每个项目不同,ProgressBar的样式需求并不一样,就算同一个项目中下拉刷新和普通加载可能也不一样,所以需要使用的小伙伴自己定义ProgressBar。
此外这套框架使用了RxCache实现缓存,而并不是通过OKHttp缓存,所以这套框架对服务器没有任何要求,不需要定义好Header之类的东西。如果你对RxCache不熟悉,可以看看这篇文章,或者RxCache的官网。不知道这么牛的框架为啥start目前才1000多呢,不说废话了,来看使用方式。
使用方式
调用以下代码完成网络请求
// 声明监听 HttpSubscriber mHttpObserver = newHttpSubscriber(newOnResultCallBack() {@OverridepublicvoidonSuccess(TestBean tb){} @OverridepublicvoidonError(intcode, String errorMsg){}});//发起请求 HttpManager.getInstance().getDatas(mHttpObserver,1,10,"json",true);//取消请求mHttpObserver.unSubscribe();
看起来是不是很简单?
下面来说说上面的代码是如何完成网络数据请求的。先来看看这个框架的结构
仅仅只有七个类而已。
简单介绍一下个各类的职责
ApiResponse——封装的返回数据模板(里面的error_code,reason,result名称需要你跟后台的小伙伴对应好,通常情况下error_code代码状态码,reason为成功或失败的提示信息,result中为具体的数据,由于数据格式未知所以使用泛型代表)
ApiService——Retrofit的数据请求接口。注意一下每个方法的返回值类型。(我们真正需要的是TestBean中的数据它必须被ApiResponse包装,最后返回Observable类型)
CacheProvider——RxCache的缓存接口,注意它的第一个参数类型必须和Retrofit数据请求接口的返回值类型一样。
OnResultCallBack——请求成功或失败的回调。
ApiException——公共的请求错误处理类。
HttpSubscriber——公共的请求订阅者。
HttpManager——发起请求的管理类。
首先是引入的各种库文件
//Rxjava2compile'io.reactivex.rxjava2:rxjava:2.0.7'compile'io.reactivex.rxjava2:rxandroid:2.0.1'//Retrofit2compile'com.squareup.retrofit2:retrofit:latest.release'compile'com.squareup.retrofit2:converter-gson:latest.release'compile'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'//RxCachecompile"com.github.VictorAlbertos.RxCache:runtime:1.8.0-2.x"compile'com.github.VictorAlbertos.Jolyglot:gson:0.0.3'//Okhttp-interceptorcompile'com.squareup.okhttp3:logging-interceptor:3.6.0'
HttpManager的初始化
publicstaticHttpManagergetInstance() {
if (instance == null) {
synchronized (HttpManager.class) {
if (instance == null) {
instance = newHttpManager();
}
}
}
returninstance;
}
//单例模式的HttpManager,来看HttpManager的构造函数
private HttpManager() {
HttpLoggingInterceptor.Level level = HttpLoggingInterceptor.Level.BODY;
HttpLoggingInterceptor loggingInterceptor = newHttpLoggingInterceptor(newHttpLoggingInterceptor.Logger()
{
@Overridepublicvoidlog(String message) {
Log.i("HttpManager", message);
}
});
loggingInterceptor.setLevel(level);
OkHttpClient.Builder builder = newOkHttpClient.Builder();
builder.connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS).retryOnConnectionFailure(true).addInterceptor(loggingInterceptor);
OkHttpClient okHttpClient = builder.build();
mRetrofit = newRetrofit.Builder().addConverterFactory(GsonConverterFactory.create()).addCallAdapterFactory(RxJava2CallAdapterFactory.create()).baseUrl(Constant.BASE_URL).client(okHttpClient).build();
cacheProvider = newRxCache.Builder().persistence(mContext.getFilesDir(), newGsonSpeaker()).using(CacheProvider.class);
mApiService = mRetrofit.create(ApiService.class);
}
在构造函数中,首先我们通过HttpLoggingInterceptor设置了拦截器,并通过addInterceptor方法设置给 OkHttpClient.Builder。然后是构建OkHttpClient.Builder及okHttpClient ,再然后是Retrofit的初始化。
下面接着是RxCache的初始化,并在其中设置了缓存目录,mContext为ApplicationContext,由init方法传入。
publicclassAppextendsApplication{@OverridepublicvoidonCreate(){super.onCreate(); HttpManager.init(this);//不做任何操作仅仅是缓存一下Application引用}}
那么HttpManager是如何完成一个网络请求的呢?
1 ApiService中声明请求接口
@FormUrlEncoded@POST("query?key=7c2d1da3b8634a2b9fe8848c3a9edcba")Observable> getDatas(@Field("pno")intpno,@Field("ps")intps,@Field("dtype") String dtype);
2CacheProvider中声明缓存接口(如果需要缓存就写)
@LifeCache(duration = 5, timeUnit = TimeUnit.MINUTES)Observable>getDatas(Observable>oRepos,EvictProvider evictDynamicKey);
注意看注解,可以自定义缓存过期时间。(EvictProvider 参数是设置是否需要对请求的数据进行缓存,具体可以看这里)同时,第一个参数一定要是Observable> 我们在ApiService中声明好的getDatas方法的返回值。
3 HttpManager声明对应的请求方法
//方法有两种,带缓存的方式
public void getDatasWithCache(Observer subscriber, intpno, intps, String dtype, booleanupdate) {
toSubscribe(cacheProvider.getDatas(mApiService.getDatas(pno, ps, dtype), newEvictProvider(update)), subscriber);
}
不带缓存的方式
public void getDatasNoCache(Observer subscriber,intpno,intps,String dtype){toSubscribe(mApiService.getDatas(pno,ps,dtype),subscriber);}
看到没不带缓存的方法只是没有用cacheProvider.getDatas()方法包裹,同时少了一个控制是否更新的参数update。
再来看一下toSubscribe()方法的实现
private void toSubscribe(Observable>o,Observer s){o.subscribeOn(Schedulers.io()).map(newFunction,T>(){@OverridepublicTapply(@NonNull ApiResponse response)throwsException{intcode=Integer.parseInt(response.getCode());if(code!=Constant.SUCCESS_CODE){thrownewApiException(code,response.getMsg());}else{if(response.getDatas()==null){return(T)response.getMsg();}else{returnresponse.getDatas();}}}}).unsubscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(s);}
借助Rxjava的map方法完成数据的进一步转换,注意SUCCESS_CODE,这个值是数据返回成功时的状态码,需要你和后台小伙伴定义好,一般情况下都为0。如果code!=SUCCESS_CODE,那么出错返回的code的和message都会抛给ApiException,在ApiException中的getApiExceptionMessage方法你可以根据具体code重新定义不同的message,或者使用传入进来的message,通常情况下这个类不需要修改,如果需要客户端根据code自定义message那么就按照上面所的方式修改即可。最后所有的message都会抛给HttpSubscriber的onError方法
@Override
public void onError(Throwable e){
if
}
(einstanceofCompositeException){
CompositeException
} compositeE=(CompositeException)e;for(Throwable throwable:compositeE.getExceptions()){if(throwableinstanceofSocketTimeoutException){mOnResultListener.onError(ApiException.Code_TimeOut,ApiException.SOCKET_TIMEOUT_EXCEPTION);}elseif(throwableinstanceofConnectException){mOnResultListener.onError(ApiException.Code_UnConnected,ApiException.CONNECT_EXCEPTION);}elseif(throwableinstanceofUnknownHostException){mOnResultListener.onError(ApiException.Code_UnConnected,ApiException.CONNECT_EXCEPTION);}elseif(throwableinstanceofRxCacheException){//缓存异常暂时不做处理}elseif(throwableinstanceofMalformedJsonException) { mOnResultListener.onError(ApiException.Code_MalformedJson,ApiException.MALFORMED_JSON_EXCEPTION); } } }else{ mOnResultListener.onError(ApiException.Code_Other,e.getMessage()); }}
在这个方法中也统一处理了网络问题。注意看不同的网络状态返回的状态码是不一样的。
SocketTimeoutException 网络超时 1000
ConnectException 链接异常 1001
UnknownHostException Host异常 1001
MalformedJsonException 解析异常 1020
其他错误信息统一返回 1003(因为code在此基本没什么用了,重要的是错误提示信息)
最终执行到
mHttpObserver=newHttpSubscriber(newOnResultCallBack(){@OverridepublicvoidonSuccess(TestBean tb){
}
@Override
public void onError(intcode,String errorMsg){
}});
中的onError方法,在这里我们根据不同的code展示不同的界面即可(例如常见的网络错误界面),或者通过Toast等其他方式给用户提示。
再回到toSubscribe方法,如果数据返回成功了,即code==SUCCESS_CODE,那么数据会返回给HttpSubscriber的onNext方法
@Override public void onNext(T t){if(mOnResultListener!=null){mOnResultListener.onSuccess(t);}}
onNext中调用OnResultCallBack的onSuccess方法,把数据传递到
mHttpObserver=newHttpSubscriber(newOnResultCallBack(){@Override public void onSuccess(TestBean tb){}
@Override public void onError(intcode,String errorMsg){}});
中的onSuccess中,在这里我们把数据展示给用户即可。
HttpSubscriber的另一项任务就是取消数据请求操作。
在onSubscribe中通过一个全局的mDisposable记录当前的Disposable
@Override public void onSubscribe(Disposable d){mDisposable=d;}
在需要取消的地方调用unSubscribe()即可
public void unSubscribe(){if(mDisposable!=null&&!mDisposable.isDisposed()){mDisposable.dispose();}}
下面运行一下程序
mHttpObserver=newHttpSubscriber(newOnResultCallBack(){
@Override public void onSuccess(TestBean tb){
for(TestBean.ListBean bean:tb.getList()){result+=bean.toString();}resultTv.setText(result);}
@Override
public void onError(intcode,String errorMsg){
resultTv.setText("onError: code:"+code+" errorMsg:"+errorMsg);
}});
这里成功后通过TextView显示一下数据,失败时也显示一下错误信息和code;
首先是不使用缓存
HttpManager.getInstance().getDatasNoCache(mHttpObserver,1,10,"json");
数据请求成功,再来测试断网情况
断网后返回了错误码1003。
下面测试缓存
设置一个5分钟的缓存
@LifeCache(duration=5, timeUnit = TimeUnit.MINUTES)Observable> getDatas(Observable> oRepos, EvictProvider evictDynamicKey);HttpManager.getInstance().getDatasWithCache(mHttpObserver,1,10,"json",false);
可以看到第一次请求产生了网络流量,说明数据来自网络,而后的几次请求均没有产生流量,说明数据来自本地缓存。
好了这套框架就介绍到这里吧,有需要的可以在这里下载
https://github.com/shiweibsw/EasyHttp
如何使用?
1 build.gradle中导入插件
2 将http包下所有内容拷贝到你的工程即可。
下一步的计划
1 retrofit 接口类封装基本的get和post请求;
2 支持以插件的形式导入工程。