Android架构设计---MVP解耦封装+组合RxJava

**版权声明:本文为小斑马伟原创文章,转载请注明出处!


上一篇介绍的是一个基础的MVP架构的登录功能。还没有进行代码的封装。它的View于Model还是有一定的耦合的。本篇目标解耦View于Model,Base基类封装,实现契约管理MVP。

一、Base抽取

类臃肿的原因是,Modle层,View层,Presenter层没有抽取,导致每一个功能模块都对应着一个M,V,P三个类。

public interface BaseView {
    void showLoading();

   void hideLoading();
}

Model层抽取

public interface BaeModel {

}

Presenter层抽取

//T代表View E代表Model
public abstract  class BasePresenter<T,E> {

public Context mContext;
public T mView;
public E mModel;
public WeakReference<T> mViewRef;

public void setViewAndModel(T view,E model) {
    mViewRef = new WeakReference<T>(view);
   this.mView = view;
    this.mModel = model;
}

public T getView() {
    if (isAttach()) {
        return mViewRef.get();
    } else {
        return null;
    }
}

public boolean isAttach() {
    return null != mViewRef && null != mViewRef.get();
}


public void onDettach() {
    if (null != mViewRef) {
        mViewRef.clear();
        mViewRef = null;
    }
}

Presenter层仍然要持有Model,View的强引用,在 setViewAndModel这个方法中,对两个对象进行赋值,这里可以看到我简单是采用了弱引用的方式去保存这个View的对象引用,减少内存泄露的可能性。
对Activity封装

二、契约管理MVP

LoginContract类将三个接口合并在一个,如下:

public class LoginContract {

interface Model extends BaeModel {
    void login(String userName, String password, UserLoginListener userLoginListener);
    Observable<String> rxLogin(String userName,String passWord);
}

interface View extends BaseView {
    void success();
    void failed();
    void clear();
}

abstract static class Presenter extends BasePresenter<View,Model> {
   abstract void login(String userName,String passWord);
   abstract void rxLogin(String userName,String passWord);
   abstract void clear();
}
}

将相同模块的M,V,P三层定一个合约,放在一块,统一管理。我们还需要看一个Activity封装的基类。

public abstract class BaseActivity<T extends  BasePresenter, E extends  BaeModel> extends AppCompatActivity {

public Context activity;
public T mPresenter;
public E mModel;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(initLayout());
    activity = this;
    mPresenter = MVPUtils.getT(this,0);
    mModel = MVPUtils.getT(this,1);

    if(null!=mPresenter) {
        mPresenter.mContext = this;
    }
    initView();
    initListener();
    initPresenter();
}

protected  abstract  void initListener();

protected  abstract  void initView();

protected  abstract  int initLayout();

protected  abstract  void initPresenter();
}

通过getGenericSuperclass方法可以获取当前对象的直接超类的 Type。

public class MVPUtils {

public static  <T> T getT(Object o, int i) {
    try {
        return  ((Class<T>) ( (ParameterizedType)(o.getClass().getGenericSuperclass())).getActualTypeArguments()[i]).newInstance();
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
    return null;
}

}

三、Model层的实现

第一个方法使用自定义回调接口实行,第二种使用RxJava方式实现

public class NewLoginModel implements LoginContract.Model {
@Override
public void login(final String userName,final String passWord,final UserLoginListener userLoginListener) {
    new Handler().postDelayed(new Runnable(){

        @Override
        public void run() {
            if("weiwei".equals(userName) && "123".equals(passWord)) {
                userLoginListener.loginSuccess();
            } else {
                userLoginListener.loginFail();
            }
        }
    },3000);
}

@Override
public Observable<String> rxLogin(final String userName,final String passWord) {
    return Observable.create(new Observable.OnSubscribe<String>() {
        @Override
        public void call(Subscriber<? super String> subscriber) {
            SystemClock.sleep(2000);
            if("weiwei".equals(userName) && "123".equals(passWord)) {
                subscriber.onNext("SUCCESS");
                subscriber.onCompleted();
            } else {
                subscriber.onNext("FAILED");
                subscriber.onCompleted();
            }
        }
    });
  }
}
三、Presenter层的实现

单纯的MVP架构在业务层是直接调用,而Android的自身不建议在主线程进行网络请求,为了保证我们业务运行于一个单独的线程,主线程只运行界面渲染,引入RxJava进行线程调度控制。具体的RxJava详细介绍见-RxJava

public class NewLoginPresenter extends LoginContract.Presenter {

@Override
void login(String userName, String passWord) {
    final LoginContract.View mView = getView();
    if(TextUtils.isEmpty(userName)||TextUtils.isEmpty(passWord)) {
        Toast.makeText(mContext,"请输入你的用户名和密码",Toast.LENGTH_SHORT).show();
    } else {
        mView.showLoading();
        mModel.login(userName, passWord, new UserLoginListener() {
            @Override
            public void loginSuccess() {
                mView.success();
                mView.hideLoading();
            }

            @Override
            public void loginFail() {
                mView.failed();
                mView.hideLoading();
            }
        });
    }
}

@Override
void rxLogin(String userName, String passWord) {
    if(TextUtils.isEmpty(userName)||TextUtils.isEmpty(passWord)) {
        Toast.makeText(mContext,"请输入你的用户名和密码",Toast.LENGTH_SHORT).show();
    } else {
        mView.showLoading();
        mModel.rxLogin(userName,passWord)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<String>() {
                    @Override
                    public void onCompleted() {
                        mView.hideLoading();
                    }

                    @Override
                    public void onError(Throwable e) {
                        mView.hideLoading();
                    }

                    @Override
                    public void onNext(String s) {
                        mView.hideLoading();
                        Toast.makeText(mContext,s,Toast.LENGTH_SHORT).show();
                    }
                });
    }
}

@Override
void clear() {
    mView.clear();
  }
}
四、Activity的实现
public class NewMVPLoginActivity extends BaseActivity<NewLoginPresenter,NewLoginModel> implements View.OnClickListener, LoginContract.View {

private EditText editAccount;
private EditText editPassWord;
private AppCompatButton btnLogin;
private AppCompatButton btnClear;
private ProgressDialog progressDialog;

@Override
protected void initListener() {
    btnLogin.setOnClickListener(this);
    btnClear.setOnClickListener(this);
}

@Override
protected void initView() {
    progressDialog = new ProgressDialog(this);
    progressDialog.setMessage("login process,Watting...");
    progressDialog.setCancelable(false);

    editAccount = (EditText)findViewById(R.id.edit_account);
    editPassWord = (EditText)findViewById(R.id.edit_password);
    btnLogin = (AppCompatButton)findViewById(R.id.btn_login);
    btnClear = (AppCompatButton)findViewById(R.id.btn_delete);
}

@Override
protected int initLayout() {
    return R.layout.activity_main;
}

@Override
protected void initPresenter() {
    mPresenter.setViewAndModel(this,mModel);
}

@Override
public void showLoading() {
    progressDialog.show();
}

@Override
public void hideLoading() {
    progressDialog.dismiss();
}

@Override
public void success() {
    Toast.makeText(NewMVPLoginActivity.this,"登录成功",Toast.LENGTH_LONG).show();
}

@Override
public void failed() {
    Toast.makeText(NewMVPLoginActivity.this,"登录失败",Toast.LENGTH_LONG).show();
}

@Override
public void clear() {
    editAccount.setText("");
    editPassWord.setText("");
}

@Override
public void onClick(View view) {
    switch(view.getId()) {
        case R.id.btn_login:
       //     mPresenter.login(editAccount.getText().toString(),editPassWord.getText().toString());
           mPresenter.rxLogin(editAccount.getText().toString(),editPassWord.getText().toString());
            break;
        case R.id.btn_delete:
            mPresenter.clear();
            break;
    }
  }
}

具体效果图如下:


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,045评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,114评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,120评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,902评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,828评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,132评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,590评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,258评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,408评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,335评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,385评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,068评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,660评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,747评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,967评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,406评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,970评论 2 341

推荐阅读更多精彩内容