Android-->MVP架构进阶(一)

相信大家一定在用mvp架构去设计App,但是在设计运用的过程中,大家有没有考虑简化代码,View层和Model层会有很多重复的代码,在显示数据之前还需每次判断View!=null,Presenter层每次需要去new Model层的实例,View层还有可能会对应多个Presenter,一个Presenter会有多个Model层等等问题。

先介绍下MVP

mvp.png
M:Model数据层:访问网络数据层都放在这里 。
V:View界面层:与View相关的一些操作都写在这里 Activity、Fragment。
P:Presenter解耦关联层(Model——View)可能还有一些额外的逻辑,数据的一些逻辑,处理业务逻辑。

View层 接收到用户操作事件,通知到Presenter,Presenter进行逻辑处理,然后通知Model更新数据,Model 把更新的数据给到Presenter,Presenter再通知到 View 更新界面。

下面是小编在设计MVP架构的项目截图

mvp.png

base包:用泛型构建基类  
HttpClient包:用Retrofit访问网络  
UserInfoContract:协议类,规范一View层、Model层、Presenter层一系列操作  
UserInfoModel:请求数据
UserInfoPresenter:连接着View层和Model层,让View层和Model层充分的解耦  

BasePresenter类

public class BasePresenter<V extends BaseView, M extends BaseModel> {
private V mView;
private M mModel;

/**
 * 绑定View
 *
 * @param view
 */
public void attach(final V view) {
    //动态代理
    mView = (V) Proxy.newProxyInstance(view.getClass().getClassLoader(), view.getClass().getInterfaces(), new InvocationHandler() {
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            //在View层显示数据之前用户可能退出了View层的页面,会在Activity的onDestroy()方法中会把mView置为null
            //由于View层都是接口,这里采用了动态代理,如果在View层显示数据之前用户可能退出了View层的页面,返回null的话,onSuccess()方法不会执行
            if (mView == null) {
                return null;
            }
            //每次调用View层接口的方法,都会执行这里
            return method.invoke(view, args);
        }
    });
    //动态创建Model?怎么创建创建???用反射创建
    Type[] params = ((ParameterizedType) this.getClass().getGenericSuperclass()).getActualTypeArguments();
    try {
        //最好是判断下类型
        mModel = (M) ((Class) params[1]).newInstance();
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
}

/**
 * 解绑View
 */
public void detach() {
    mView = null;
    mModel = null;
}

public V getView() {
    return mView;
}

public M getModel() {
    return mModel;
}

利用泛型动态去构建View层、Model层。Model层创建实例用反射。BasePresenter会持有View层、Model层的引用,由于View都是接口,Activity会实现这个接口的所有方法,Presenter又会去调用这些方法,所以这里采用了动态代理,如果View==null的话(在成功显示数据之前用户可能会退出页面,在onDestory()方法中会解绑View,View会被置为null)不会执行method.invoke()方法,也就是不会执行向服务器请求数据成功的方法。

BaseMvpActivity

public abstract class BaseMvpActivity<P extends BasePresenter> extends AppCompatActivity implements BaseView {
    private P mPresenter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(getLayoutId());
        //创建Presenter,交给子类去实现
        mPresenter = createPresenter();
        //让P层去绑定V
        mPresenter.attach(this);
        initView();
        initData();

    }

    protected abstract P createPresenter();

    protected abstract int getLayoutId();

    protected abstract void initView();

    protected abstract void initData();

    public P getPresenter() {
        return mPresenter;
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mPresenter.detach();
}

BaseMvpActivity作为基类。

UserInfoContract类 负责定义管理接口

    public class UserInfoContract {
    //View层:视图层即Activity/Fragment
    public interface UserInfoView extends BaseView {
        void onLoading();

        void onSuccess(User user);

        void onError(int code, String errorMessage);

    }

    //Presenter层:处理业务逻辑
    interface UserInfoPresenter {
      void getUserInfo(String userName);
    }

    //Model层,处理数据层,比如请求网络,操作数据库等,外部只需关心Model返回的数据,无需关心内部细节
    interface UserModel {
       void getUserInfo(String userName,HttpCallback<User> httpCallback);
    }
}

UserInfoContract:协议类,规范一View层、Model层、Presenter层一系列操作。
UserInfoModel类 model层处理数据层

public class UserInfoModel extends BaseModel implements UserInfoContract.UserModel {

    @Override
    public void getUserInfo(String userName, final HttpCallback<User> httpCallback) {
        RetrofitClient.getServiceApi().getUser(userName).enqueue(new Callback<BaseResult<User>>() {
            @Override
            public void onResponse(@NonNull Call<BaseResult<User>> call, @NonNull Response<BaseResult<User>> response) {
                BaseResult<User> userBaseResult = response.body();
                if (response.isSuccessful() && userBaseResult != null) {
                    if (userBaseResult.getCode() == Constant.SUCCESS_CODE && userBaseResult.getData() != null) {
                        httpCallback.onSuccess(userBaseResult.getData());
                    } else {
                        httpCallback.onFail(userBaseResult.getCode(), userBaseResult.getMsg());
                    }
                }
            }

            @Override
            public void onFailure(@NonNull Call<BaseResult<User>> call, @NonNull Throwable t) {
                httpCallback.onFail(Constant.SERVER_EXCEPTION, t.getMessage());

            }
        });

    }
}

UseInfoModel负责请求网络,处理数据层。

UserInfoPresenter 类 Presenter层处理业务逻辑

public class UserInfoPresenter extends BasePresenter<UserInfoContract.UserInfoView, UserInfoModel> implements UserInfoContract.UserInfoPresenter {
    @Override
    public void getUserInfo(String userName) {
        getView().onLoading();
        getModel().getUserInfo(userName, new HttpCallback<User>() {
            @Override
            public void onSuccess(User user) {
                getView().onSuccess(user);

            }

            @Override
            public void onFail(int code, String error) {
                getView().onError(code, error);
            }
        });

    }
    }

UserInfoPresenter是连接UserInfoView和UseInfoModel的桥梁,也充分体现了View层和Model层解耦。实现了业务逻辑代码。

MainActivity类

public class MainActivity extends BaseMvpActivity<UserInfoPresenter> implements UserInfoContract.UserInfoView {
private TextView mTextView;
/**
 * 创建Presenter
 *
 * @return
 */
@Override
protected UserInfoPresenter createPresenter() {
    return new UserInfoPresenter();
}

/**
 * 返回Activity的布局Id
 *
 * @return
 */
@Override
protected int getLayoutId() {
    return R.layout.activity_main;
}

/**
 * 初始化View
 */
@Override
protected void initView() {
    mTextView = findViewById(R.id.tv);

}

/**
 * 在这里去请求数据
 */
@Override
protected void initData() {
    getPresenter().getUserInfo("Steven");
}

/**
 * 显示一个加载的进度条
 */
@Override
public void onLoading() {

}

/**
 * 请求数据成功回调该方法
 *
 * @param user
 */
@Override
public void onSuccess(User user) {
    mTextView.setText("Hello:" + user.getData().getUserName());
}

/**
 * 请求数据失败回调该方法
 */
@Override
public void onError() {

}

这个Simple的完整代码地址:https://github.com/StevenYan88/AndroidMvp.git

以上解决了View层和Model层会有很多重复的代码,在显示数据之前还需每次判断View!=null,Presenter层每次需要去new Model层的实例这几个问题,View层还有可能会对应多个Presenter,一个Presenter会有多个Model层这几个问题还没有解决。

MVP架构进阶(二)https://www.jianshu.com/p/6326fa62b8aa

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

推荐阅读更多精彩内容