Android---MVP模式---大道至简

一. 前言

看了很多关于MVP的文章,有初级篇,高级篇,还有终极篇 等等,给我的感觉就是: 有没有必要封装的那么复杂?

1. MVP主要解决了什么问题?

当页面比较复杂时: Activity 处理的东西太多,太杂,所以得按一定的规则分离出来.

2. 引发常见的问题?

① 内存泄露:
因为P层持有view层的引用, 目前常见的封装方式,基本都解决了这个问题。
② 新增的类过多:
例如:登陆页,ILoginView, LoginP, ILoginP(用于M层回调,非必须), loginM。
每个页面都要加这几个类, 反正我是写的挺烦的, 也有的兄弟把这个弄成插件,自动生成.
③一个V对应多个P(P层复用), 不太友好:
④ 学习成本相对比较高:
特别是:一些高级的封装方式,比如动态代理.
使用上: 需要关注的类和方法比较多,而且要一一对应.
封装逻辑比较复杂: 如果别人需要修改,花费时间相对会多一些.

正因为有了以上的问题,所以才有了后面MVVM模式. 这里就不说了.
MVP解决方案如下:

二. 先最终调用效果

public class LoginActivity extends MvpActivity<LoginPresenter>{
    //...布局等代码忽略
    @Override
    public void init() {
        presenter.login();  //登录
    }

    @Override
    public void mvpSuccess(Object data, String type) {  
        //1.只有一个接口的话,直接强转对象
       //2.多个接口回调的话, 根据type判断,强转对象
    }
}
public class LoginPresenter extends MvpPresenter{

    public LoginPresenter(ImvpView imvpView) {
        super(imvpView);
    }

    public void login(){  //.....请求
        //成功
        mvpSuccess(new LoginResult("回调对象"), "loginType");
        //失败
        mvpFail(0, "登陆失败", "loginType");
    }
}
说明:

1. 一个Presenter,一个接口的话, mvpSuccess回调,直接强转对象
2. 一个Presenter,多个接口的话, mvpSuccess回调, 根据返回的type强转对象.
3. 多个Presenter. 如下:mvpSuccess回调,也是根据type强转对象.

public class LoginActivity extends MvpActivity<LoginPresenter>{

    TestPresenter testPresenter;  //其它的 Presenter

    @Override
    public void init() {
        testPresenter = new TestPresenter(this);
        testPresenter.test(); //发起请求
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if(testPresenter!= null){ //记得解绑
            testPresenter.unbind();
        }
    }
}

1. 大伙觉得这样的使用方式怎么样呢? 虽然强转类型感觉不太友好,但是总体我觉得方便了很多.
2. 看到这里,可能还有很多疑问,例如,部分页面失败回调时候需要单独处理怎么办?

三. 封装过程(主要看V和P层, M层一样的道理)

1. 总的回调接口ImvpView
具体方法根据个人需要自定义, 例如:showLoading(String type), dissmiss(String type).

//回调接口
public interface ImvpView {
    //成功
    void mvpSuccess(Object data, String type);
    //失败
    void mvpFail(int code, String msg, String type);
}

2.MvpActivity类:
①: 抽象类,实现回调接口ImvpView.
②: 重写抽象方法,这样Activity页面需要什么方法就重写什么方法即可.
③: 定义一个泛型变量P, 这样无需每个Activity都定义变量P,P的初始化可以用反射, 也可以在具体Activity中new对象.
④: onDestroy解绑.

public abstract class MvpActivity<T extends MvpPresenter> extends BaseActivity implements ImvpView{

    public T presenter;

    public void layoutBefore(){
        setPresenter();
    }

    private void setPresenter(){
            //1. 懒的话可用反射直接创建, 子类无需再new
            //2. 追求性能, 而且不觉得麻烦的话,可以子类new对象.
    }

    @Override
    public void mvpFail(int code, String msg, String type) {}

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if(presenter != null){
            presenter.unbind();
        }
    }

}
public abstract class BaseActivity extends AppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        layoutBefore();
        setContentView(getLayoutId());
        init();
    }
    public void layoutBefore(){}
    public abstract int getLayoutId();
    public abstract void init();

}

3.MvpPresenter类:
①: 主要思想, 提供给子类,调用ImvpView 里的方法,这个类只是加上的非空判断.
②: 提供方法:解除与Activity的关联。
③: 可扩展自己需要的方法,不会影响其它地方.

public class MvpPresenter{

    ImvpView imvpView;
    String defaultType = ""; //返回类型

    public MvpPresenter(ImvpView imvpView){
        this.imvpView = imvpView;
    }

    protected void mvpSuccess(Object data, String type){  //成功
        if(imvpView == null){
            return;
        }
        imvpView.mvpSuccess(data, type);
    }

    protected void mvpFail(int code, String msg, String type){ //失败
        if(imvpView == null){
            return;
        }
        imvpView.mvpFail(code, msg, type);
    }


    /**
     * 解绑
     */
    public void unbind(){
        imvpView = null;
    }
}

四: 总结

1. 学习成本低,使用简单, 封装逻辑简单,小白应该都能看懂.
调用时: 耦合性低了很多,V层只需关心需要哪些P, P层只需专心拿数据, 完全不需要关注其它类.
2. 解决了: 内存泄露,P的复用, 类过多 的问题
3. 扩展性: 在ImvpView, MvpActivity, MvpPresenter三个核心类, 新增功能, 其它地方完全不受影响.
4.不足: 强转类型感觉不太友好.

我做过的项目中,觉得这样封装暂时足够用了, 如有不足,欢迎指出。

最后附上: 反射初始化P的代码:

 /**
     * 初始化Presenter(觉得麻烦可忽略, 因为直接 new即可)
     */
    private void setPresenter(){
        try {
            Type superClass = getClass().getGenericSuperclass();
            Type type = ((ParameterizedType) superClass).getActualTypeArguments()[0];
            Class<?> clazz = getRawType(type);
            Constructor constructor =clazz.getConstructor(ImvpView.class);
            presenter = (T) constructor.newInstance(this);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    // type不能直接实例化对象,通过type获取class的类型,然后实例化对象
    public static Class<?> getRawType(Type type) {
        if (type instanceof Class) {
            return (Class) type;
        } else if (type instanceof ParameterizedType) {
            ParameterizedType parameterizedType = (ParameterizedType) type;
            Type rawType = parameterizedType.getRawType();
            return (Class) rawType;
        } else if (type instanceof GenericArrayType) {
            Type componentType = ((GenericArrayType) type).getGenericComponentType();
            return Array.newInstance(getRawType(componentType), 0).getClass();
        } else if (type instanceof TypeVariable) {
            return Object.class;
        } else if (type instanceof WildcardType) {
            return getRawType(((WildcardType) type).getUpperBounds()[0]);
        } else {
            String className = type == null ? "null" : type.getClass().getName();
            throw new IllegalArgumentException("Expected a Class, ParameterizedType, or GenericArrayType, but <" + type + "> is of type " + className);
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,692评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,482评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,995评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,223评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,245评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,208评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,091评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,929评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,346评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,570评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,739评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,437评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,037评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,677评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,833评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,760评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,647评论 2 354

推荐阅读更多精彩内容