在前面已经初步封装了一个MVP的网络请求框架,那只是个雏形,还有很多功能不完善,现在进一步进行封装。添加了网络请求时的等待框,retrofit中添加了日志打印拦截器,添加了token拦截器,并且对DataManager类进行了扩展,真正体现它的作用,并且对大量的重复代码做了一定封装,减少代码的冗余。

下面结合上篇文章,进行下一步的封装。

1、首先完善Result.java这个类。

通常在我们写API接口文档的时候,后端返回的数据格式都是

"code":1    //1:成功

//-1:token验证失败

“msg”:”success”, //返回的消息提示

“token验证失败”

“data”:   //数据

{

“username”:” xdw” ,  //用户名

"age":30  //年龄

}

具体的Result.java的代码如下,里面还加入了一个对返回码的判断方法

package com.xdw.retrofitrxmvpdemo.model;

import com.xdw.retrofitrxmvpdemo.constant.Constant;

/**
* Created by 夏德旺 on 2017/12/8.
*/ public class Result<T> {
private int code;
private String msg;
private T data; public Result(int code, String msg, T data) {
this.code = code;
this.msg = msg;
this.data = data;
}
//添加对返回状态成功的判断
public boolean isSuccess() {
return code == Constant.SUCCESS;
} public int getCode() {
return code;
} public String getMsg() {
return msg;
} public T getData() {
return data;
} }

2、添加ProgressDialogHandler和ProgressCancelListener,用来处理网络请求等待框。代码如下

ProgressCancelListener:

package com.xdw.retrofitrxmvpdemo.model;
/**
* Created by 夏德旺 on 2017/12/8.
*/
public interface ProgressCancelListener {
void onCancelProgress();
}
 

ProgressDialogHandler:

package com.xdw.retrofitrxmvpdemo.http;

import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.os.Handler;
import android.os.Message; /**
* Created by 夏德旺 on 2017/12/8.
*/
public class ProgressDialogHandler extends Handler { public static final int SHOW_PROGRESS_DIALOG = 1;
public static final int DISMISS_PROGRESS_DIALOG = 2; private ProgressDialog pd; private Context context;
private boolean cancelable;
private boolean show;
private ProgressCancelListener mProgressCancelListener; public ProgressDialogHandler(Context context, ProgressCancelListener mProgressCancelListener,
boolean cancelable,boolean show) {
super();
this.context = context;
this.mProgressCancelListener = mProgressCancelListener;
this.cancelable = cancelable;
this.show = show;
} private void initProgressDialog(){
if (pd == null) {
pd = new ProgressDialog(context); pd.setCancelable(cancelable); if (cancelable) {
pd.setOnCancelListener(new DialogInterface.OnCancelListener() {
@Override
public void onCancel(DialogInterface dialogInterface) {
mProgressCancelListener.onCancelProgress();
}
});
} if (!pd.isShowing()&&show) {
pd.show();
}
}
} private void dismissProgressDialog(){
if (pd != null) {
pd.dismiss();
pd = null;
}
} @Override
public void handleMessage(Message msg) {
switch (msg.what) {
case SHOW_PROGRESS_DIALOG:
initProgressDialog();
break;
case DISMISS_PROGRESS_DIALOG:
dismissProgressDialog();
break;
}
} }

3、改写RetrofitApiService,将返回结果由原来的UserInfo改为Result<UserInfo>。

public interface RetrofitApiService {

    @GET("user")
Observable<Result<UserInfo>> getUserInfo(@Query("uid") int uid); }

4、完善之前的RetrofitUtil,加入日志与token拦截器,token这段我注释掉了,根据自己的实际项目进行添加

package com.xdw.retrofitrxmvpdemo.http;

import android.content.Context;

import com.google.gson.GsonBuilder;
import com.xdw.retrofitrxmvpdemo.BuildConfig;
import com.xdw.retrofitrxmvpdemo.constant.UrlConstant; import java.io.IOException; import okhttp3.Interceptor;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.logging.HttpLoggingInterceptor;
import retrofit2.Retrofit;
import retrofit2.adapter.rxjava.RxJavaCallAdapterFactory;
import retrofit2.converter.gson.GsonConverterFactory; /**
* Created by 夏德旺 on 2017/12/8.
*/ public class RetrofitUtil {
private Context mCntext;
//声明Retrofit对象
private Retrofit mRetrofit;
//声明RetrofitApiService对象
private RetrofitApiService retrofitApiService;
GsonConverterFactory factory = GsonConverterFactory.create(new GsonBuilder().create());
//由于该对象会被频繁调用,采用单例模式,下面是一种线程安全模式的单例写法
private volatile static RetrofitUtil instance; public static RetrofitUtil getInstance(Context context){
if (instance == null) {
synchronized (RetrofitUtil.class) {
if (instance == null) {
instance = new RetrofitUtil(context);
}
}
}
return instance;
}
private RetrofitUtil(Context mContext){
mCntext = mContext;
init();
} //初始化Retrofit
private void init() { //添加token拦截
/* final String token = AppSPUtils.getValueFromPrefrences(AppConstants.SP_TOKEN, "");
final int uid = AppSPUtils.getValueFromPrefrences(AppConstants.SP_USERID, 0);
Interceptor mTokenInterceptor = new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request authorised = chain.request().newBuilder()
.addHeader("userId", String.valueOf(uid))
.addHeader("token", token)
.build();
return chain.proceed(authorised);
}
};*/
//打印请求log日志
OkHttpClient httpClient = new OkHttpClient();
if (BuildConfig.DEBUG) {
HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor();
httpLoggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
httpClient = new OkHttpClient.Builder()
.addInterceptor(httpLoggingInterceptor)
// .addInterceptor(mTokenInterceptor)
.build();
}
mRetrofit = new Retrofit.Builder()
.baseUrl(UrlConstant.BASE_URL)
.client(httpClient)
.addConverterFactory(factory)
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
retrofitApiService = mRetrofit.create(RetrofitApiService.class);
} public RetrofitApiService getRetrofitApiService(){
return retrofitApiService;
}
}

5、重要的地方来了,完善之前的DataManager。之前这个类大家可以觉得非常鸡肋,因为它什么也没干,就是把RetrofitApiService和RetrofitUtil 该干的活移动到了这类中,非但没有减轻任务量,反而要多写一大堆重复代码。等现在封装之后就可以发现它的大作用了。

我们现在在RetrofitApiService中的getUserInfo方法的返回值变成了Result<UserInfo>,但是实际上最后我们要的数据仅仅是UserInfo,这时可以对之前的DataManager中的getUserInfo方法修改下,如下

    public Observable<UserInfo>  getUserInfo(int uid){
return mRetrofitService.getUserInfo(uid).map(new ResultFunc<UserInfo>() );
}

这里使用了rxjava中的map这个关键方法将数据进行了剥离出来,不懂这个方法的请自己去查阅rxjava的资料。

在DataManager中同时定义了一个内部类,如下

 public class ResultFunc<T> implements Func1<Result<T>, T> {
@Override
public T call(Result<T> result) {
//在这里对对服务端返回的resultCode进行判断,如果返回码不是成功,
// 则抛出自定义的API异常,比如token登陆失败时。抛出异常的异常可以统一
// 在Subscriber的子类ProgressSubscriber中进行处理,这样就不用到
// 每个Activity中再来进行处理
if (!result.isSuccess()) {
throw new APIException(result.getCode(), result.getMsg());
}
return result.getData();
}
}

这里补充自己定义的一个异常类APIException的代码,如下

package com.xdw.retrofitrxmvpdemo.util;

/**
* 自定义异常
* Created by 夏德旺 on 2017/12/8.
*/
public class APIException extends RuntimeException{
public int code;
public String message; public APIException(int code, String message) {
this.code = code;
this.message = message;
} @Override
public String getMessage() {
return message;
} public int getCode() {
return code;
}
}

DataManager的封装到此完成,具体它的应用请看后面的UserInfoPresenter中的调用

6、在BasePresenter中添加一个addSubscription方法,这个是对每次的订阅进行封装,简化重复代码量

    //将每次的订阅操作进行封装,简化重复代码量
public <T> void addSubscription(Observable<T> o, Subscriber<T> s) {
mCompositeSubscription.add(o.unsubscribeOn(Schedulers.io())
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(s)); }

7、再来看看具体处理业务逻辑的UserInfoPresenter的代码是不是会清爽很多

package com.xdw.retrofitrxmvpdemo.presenter;

import android.content.Context;
import android.util.Log; import com.xdw.retrofitrxmvpdemo.http.ProgressSubscriber;
import com.xdw.retrofitrxmvpdemo.manager.DataManager;
import com.xdw.retrofitrxmvpdemo.model.Result;
import com.xdw.retrofitrxmvpdemo.model.UserInfo;
import com.xdw.retrofitrxmvpdemo.pv.PresentView;
import com.xdw.retrofitrxmvpdemo.pv.UserInfoPv; import rx.Observable;
import rx.Observer;
import rx.Subscriber;
import rx.android.schedulers.AndroidSchedulers;
import rx.functions.Func1;
import rx.schedulers.Schedulers; /**
* Created by 夏德旺 on 2017/12/8.
*/ //该类是具体业务presenter,如需增加另一个业务,比如Order
//则可以再创建一个OrderPresenter
public class UserInfoPresenter extends BasePresenter {
private Context mContext;
private UserInfoPv mUserInfoPv;
private UserInfo mUserInfo; public UserInfoPresenter(Context context) {
this.mContext = context;
} @Override
public void BindPresentView(PresentView presentView) {
mUserInfoPv = (UserInfoPv) presentView;
} //在presenter中实现业务逻辑,此处会调用前面封装好的retrofit的东西
//将处理结果绑定到对应的PresentView实例,这样Activity和PresentView实例绑定好之后,
//Activity->PresentView->Presenter->retrofit的关系就打通了
public void getUserInfo(int uid) {
Observable<UserInfo> observable = DataManager.getInstance(mContext).getUserInfo(uid);
addSubscription(observable,new ProgressSubscriber<UserInfo>(mUserInfoPv, mContext, true) {
@Override
public void onNext(UserInfo userInfo) {
super.onNext(userInfo);
mUserInfoPv.onSuccess(userInfo);
}
} );
}
}

这个getUserInfo是不是清爽了很多,首先通过DataManager的封装,可以很方便的获取剥离出了核心数据的observable,然后调用父类中的addSubscription来处理具体的业务。大家看到这里会发现业务逻辑中的onCompleted与onError这2个核心方法怎么不见了。这些我们都统一封装到了ProgressSubscriber中。继续往下看

8、封装ProgressSubscriber,它继承Subscriber类,并且实现ProgressCancelListener接口。在该类中统一对报错(并且包括自定义的API异常)和网络对话框的出现与消失做了处理。这样就简化了我们大量的操作,不用再到每个界面中单独对其进行判断处理。具体代码如下

package com.xdw.retrofitrxmvpdemo.http;
import android.content.Context;
import android.util.Log;
import android.widget.Toast; import com.xdw.retrofitrxmvpdemo.pv.PresentView;
import com.xdw.retrofitrxmvpdemo.util.APIException; import java.net.ConnectException;
import java.net.SocketTimeoutException; import rx.Subscriber; /**
* 用于在Http请求开始时,自动显示一个ProgressDialog
* 在Http请求结束时,关闭ProgressDialog
* 调用者自己对请求数据进行处理
* Created by 夏德旺 on 2017/12/8.
*/
public class ProgressSubscriber<T> extends Subscriber<T> implements ProgressCancelListener{ private PresentView mPresentView;
private ProgressDialogHandler mProgressDialogHandler; private Context context; public ProgressSubscriber(PresentView mPresentView, Context context, boolean show) {
this.mPresentView = mPresentView;
this.context = context;
mProgressDialogHandler = new ProgressDialogHandler(context, this, true,show);
}
public ProgressSubscriber(Context context,boolean show) {
this.context = context;
mProgressDialogHandler = new ProgressDialogHandler(context, this, true,show);
} private void showProgressDialog(){
if (mProgressDialogHandler != null) {
mProgressDialogHandler.obtainMessage(ProgressDialogHandler.SHOW_PROGRESS_DIALOG).sendToTarget();
}
} private void dismissProgressDialog(){
if (mProgressDialogHandler != null) {
mProgressDialogHandler.obtainMessage(ProgressDialogHandler.DISMISS_PROGRESS_DIALOG).sendToTarget();
mProgressDialogHandler = null;
}
} /**
* 订阅开始时调用
* 显示ProgressDialog
*/
@Override
public void onStart() {
showProgressDialog();
} /**
* 完成,隐藏ProgressDialog
*/
@Override
public void onCompleted() {
dismissProgressDialog();
} /**
* 对错误进行统一处理
* 隐藏ProgressDialog
* @param e
*/
@Override
public void onError(Throwable e) {
if (e instanceof SocketTimeoutException) {
Toast.makeText(context,"网络中断,请检查您的网络状态",Toast.LENGTH_SHORT).show();
} else if (e instanceof ConnectException) {
Toast.makeText(context,"网络中断,请检查您的网络状态",Toast.LENGTH_SHORT).show();
} else if(e instanceof APIException){
Toast.makeText(context,e.getMessage(),Toast.LENGTH_SHORT).show();
Log.e("xdw","apiCode="+((APIException) e).getCode());
}else{
Toast.makeText(context,"未知异常",Toast.LENGTH_SHORT).show();
Log.e("xdw","apiCode="+((APIException) e).getCode());
}
dismissProgressDialog(); } /**
* 将onNext方法中的返回结果交给Activity或Fragment自己处理
*
* @param t 创建Subscriber时的泛型类型
*/
@Override
public void onNext(T t) { } /**
* 取消ProgressDialog的时候,取消对observable的订阅,同时也取消了http请求
*/
@Override
public void onCancelProgress() {
if (!this.isUnsubscribed()) {
this.unsubscribe();
}
}
}

9、由于已经对错误进行了统一处理,那么这里在PresentView接口中去掉了之前定义的onError方法。

package com.xdw.retrofitrxmvpdemo.pv;

/**
* Created by 夏德旺 on 2017/12/8.
*/ public interface PresentView{
//定义一个最基础的接口,里面就包含一个出错信息的回调
//因为大多数时候报错的时候都是采用一条信息提示
//如果需要负责的报错接口,请重载onError,是重载不是重写
//经过后面加入第二次封装之后,这里的onError删除了,统一封装到了ProgressSubscriber中
//如果需要特定的具体要到每个Activity中去重写报错信息的,可以在此再添加一个onError处理
}

10、最后我们来看看MainActivity中的处理,基本和之前没什么变化,就是少了个onError的方法

package com.xdw.retrofitrxmvpdemo.activity;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView; import com.xdw.retrofitrxmvpdemo.R;
import com.xdw.retrofitrxmvpdemo.model.UserInfo;
import com.xdw.retrofitrxmvpdemo.presenter.UserInfoPresenter;
import com.xdw.retrofitrxmvpdemo.pv.UserInfoPv; public class MainActivity extends AppCompatActivity {
private TextView text;
private Button button;
//定义需要调用的presenter对象
private UserInfoPresenter mUserInfoPresenter =new UserInfoPresenter(this); @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
text = (TextView)findViewById(R.id.text);
button = (Button)findViewById(R.id.button);
//在Activity创建的时候同时初始化presenter,这里onCreater不是指的创建presenter对象,
// 而是做一些presenter的初始化操作,名字应该取名init更好理解点,我这里就不重命名了
mUserInfoPresenter.onCreate();
//将presenter和PresentView进行绑定,实际上就是将presenter和Activity视图绑定,
//这个是MVP模式中V与P交互的关键
mUserInfoPresenter.BindPresentView(mUserInfoPv);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//点击按钮触发presenter里面的方法
mUserInfoPresenter.getUserInfo(1);
}
}); } //采用内部类方法定义presentView对象,该对象用来将Activity和presenter进行绑定
//绑定了以后主线程中就可以通过回调来获取网络请求的数据
private UserInfoPv mUserInfoPv = new UserInfoPv(){
@Override
public void onSuccess(UserInfo userInfo) {
text.setText(userInfo.toString());
}
}; //在Activity销毁的时候,一定要对CompositeSubscription进行释放,否则会造成内存泄漏
//释放操作封装到了presenter的ondestroy方法中
@Override
protected void onDestroy(){
super.onDestroy();
mUserInfoPresenter.onDestroy();
}
}

到此整个封装完毕,我也用自己之前做的项目检验了下,确实也很好用。下面跟上篇博客一样介绍下使用方法。

列举下之后像该项目中扩展业务的步骤,比如加一个订单功能。

操作步骤:

1、添加对应的model类Order

2、RetrofitApiService中添加对应的网络请求api,此时的api格式是带上Result的

3、将新添加的api映射到DataManager中,此时在DataManager中的api是剥离出实际数据之后的

4、添加业务对应的PrensentView实例OrderPv

5、添加业务对应的Presenter实例OrderPresenter

6、在需要该业务的UI线程(Activity或Fragment)中调用具体业务对应的Presenter

其实操作步骤没有太大变化,但是将返回结果换成了统一的数据格式,加入了返回码和返回消息,方便客户端对错误进行统一处理。同时添加了log与token的拦截器,并且简化了RxJava相关的代码操作。至此,一个完整的MVP框架封装完成。

一步步搭建Retrofit+RxJava+MVP网络请求框架(二),个人认为这次封装比较强大了的更多相关文章

  1. 一步步搭建Retrofit+RxJava+MVP网络请求框架(一)

    首先,展示一下封装好之后的项目的层级结构. 1.先创建一个RetrofitApiService.java package com.xdw.retrofitrxmvpdemo.http; import ...

  2. android开发学习 ------- Retrofit+Rxjava+MVP网络请求的实例

    http://www.jianshu.com/p/7b839b7c5884   推荐 ,照着这个敲完 , 测试成功 , 推荐大家都去看一下 . 下面贴一下我照着这个敲完的代码: Book实体类 - 用 ...

  3. kotlin for android----------MVP模式下(OKHttp和 Retrofit+RxJava)网络请求的两种实现方式

    今天要说的干货是:以Kotlin,在MVP模式下(OKHttp和 Retrofit+RxJava)网络请求两种实现方式的一个小案例,希望对大家有所帮助,效果图: Retrofit是Square公司开发 ...

  4. 基于Retrofit+RxJava的Android分层网络请求框架

    目前已经有不少Android客户端在使用Retrofit+RxJava实现网络请求了,相比于xUtils,Volley等网络访问框架,其具有网络访问效率高(基于OkHttp).内存占用少.代码量小以及 ...

  5. Android 网络请求框架Retrofit

    Retrofit是Square公司开发的一款针对Android网络请求的框架,Retrofit2底层基于OkHttp实现的,OkHttp现在已经得到Google官方认可,大量的app都采用OkHttp ...

  6. 安卓开发常用网络请求框架OkHttp、Volley、XUtils、Retrofit对比

    网络请求框架总结1.xutils     此框架庞大而周全,这个框架可以网络请求,同时可以图片加载,又可以数据存储,又可以 View 注解,使用这种框架很方便,这样会使得你整个项目对它依赖性太强,万一 ...

  7. 【转载】一步一步搭建自己的iOS网络请求库

    一步一步搭建自己的iOS网络请求库(一) 大家好,我是LastDay,很久没有写博客了,这周会分享一个的HTTP请求库的编写经验. 简单的介绍 介绍一下,NSURLSession是iOS7中新的网络接 ...

  8. App 组件化/模块化之路——如何封装网络请求框架

    App 组件化/模块化之路——如何封装网络请求框架 在 App 开发中网络请求是每个开发者必备的开发库,也出现了许多优秀开源的网络请求库.例如 okhttp retrofit android-asyn ...

  9. Android网络请求框架

    本篇主要介绍一下Android中经常用到的网络请求框架: 客户端网络请求,就是客户端发起网络请求,经过网络框架的特殊处理,让后将请求发送的服务器,服务器根据 请求的参数,返回客户端需要的数据,经过网络 ...

随机推荐

  1. LeetCode 665. Non-decreasing Array (不递减数组)

    Given an array with n integers, your task is to check if it could become non-decreasing by modifying ...

  2. 教你用SVG画出一条龙

    先看demo,九十七度 其实使用svg画出这条龙很简单,关键不在于怎么使用svg,而在于你的美术功底,哈哈. 好吧,当然基础是不能忽略的,先看下这条龙的代码: <svg id="lon ...

  3. Java基础笔记11

    异常: 即java程序在运行时出现的意外情况.  java如何处理异常. try{ //可能发生异常的地方 }catch(异常类型 对象){  //异常处理处 }catch(异常类型 对象){ }.. ...

  4. Linux下Git安装、配置

    安装 首先查看下有没有安装过 输入 git,出现以下的,就说明安装过了. 否则, 执行命令:sudo apt-get install git    进行安装 安装好之后,还需要执行命令: git co ...

  5. zookeeper启动异常

    zookeeper启动报异常 java.io.EOFException  at java.io.DataInputStream.readInt(DataInputStream.java:392) 遇到 ...

  6. HDU 6047 Maximum Sequence

    Maximum Sequence Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  7. C++求出旋转数组的最小数字

    今天遇到这么一道题目,感觉很有意思,要记下来! 题目:把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转. 输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素. 例如数组{3,4 ...

  8. Python filter用法

    class filter(object) | filter(function or None, iterable) --> filter object | | Return an iterato ...

  9. filereader api 类型

    filereader类似XMLHttpRequest,只是它用来从文件系统读取文件,提供了不同的方法去读取文件数据:1.readAsText2.readAsDataURL3.readAsBinaryS ...

  10. 按键精灵 vbs 获取网页源码 xp系统被拒绝

    如下面的代码所示,获取新浪博客某个指定网页的源码 verurl = "http://blog.sina.com.cn/s/blog_9ea1db7b0101o7ch.html?" ...