史上最最最简单的MVP教程

史上最最最简单的MVP教程


作者 我是吴长老啊
2016.09.27 16:51* 字数 1188 阅读 3397 评论 60

本篇文章适合那些对MVP设计模式还不是很熟悉的同学。如果有写得不对的地方,欢迎指正。

这几天一直加班,好不容易到了周末,就狂睡一通,结果白天睡觉,晚上反而睡不着了。闲着也么事,就起来写写博客吧,记录一下最近工作中遇到的问题。

刚进新公司的时候,接手了公司的项目,可是刚拿到代码就蒙了,完全看不懂。。。新公司用的是MVP架构,但是我之前的公司用的是MVC架构,有的人可能就会问了,为了么MVP架构这么流行,还有公司不去用它,这个我也没办法,也不是我们今天讨论的话题。

拿着公司的代码,边做功能边学习MVP,这个还是比较痛苦的,刚开始什么都看不懂,方法在哪里被调用的也看不出来,只能慢慢摸索,好在这两天的努力没白费,终于把MVP弄懂了,下面我就以我的思路来解释一下什么叫MVP。

我们就写一个简单的Demo吧,从实例中来学习比较快。网上大部分的MVP例子都是登录注册,我也来写一个登录注册。

首先,我们需要定义一个资源文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
    xmlns:tools="http://schemas.android.com/tools"  
    android:layout_width="match_parent"  
    android:layout_height="match_parent"  
    android:orientation="vertical">  

    <EditText  
        android:id="@+id/username_et"  
        android:hint="请输入账号"  
        android:layout_width="match_parent"  
        android:layout_height="wrap_content" />  

    <EditText  
        android:id="@+id/password_et"  
        android:hint="请输入密码"  
        android:layout_width="match_parent"  
        android:layout_height="wrap_content"  
        android:inputType="textPassword" />  

    <Button  
        android:id="@+id/login_btn"  
        android:layout_width="wrap_content"  
        android:layout_height="wrap_content"  
        android:text="登录"/>  

</LinearLayout>

代码很简单,一个账号输入框,一个密码输入框,一个登录按钮

然后看一下,不使用MVP模式的MainActivity代码:

public class MainActivity extends Activity implements OnClickListener {  

    private static final String TAG = "MainActivity";  
    public static final String USERNAME="wuzhanglao";  
    public static final String PASSWORD="470782682";  

    private EditText username_et;  
    private EditText password_et;  
    private Button login_btn;  

    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_main);  
        initView();  
    }  

    private void initView() {  
        username_et = (EditText) findViewById(R.id.username_et);  
        password_et = (EditText) findViewById(R.id.password_et);  
        login_btn = (Button) findViewById(R.id.login_btn);  
        login_btn.setOnClickListener(this);  
    }  

    @Override  
    public void onClick(View v) {  
        switch (v.getId()) {  
        case R.id.login_btn:  
            String username = username_et.getText().toString();  
            String password = password_et.getText().toString();  
            if(username.isEmpty() || password.isEmpty()){  
                Log.d(TAG, "账号或者密码不能为空");  
            } else if(username.length() <8 || password.length()<9){  
                Log.d(TAG, "账号至少8位,密码至少10位");  
            } else if(username.equals(USERNAME)){  
                if(password.endsWith(PASSWORD)){  
                    Log.d(TAG, "恭喜你,登录成功");  
                } else {  
                    Log.d(TAG, "登录失败->密码错误");  
                }  
            }  
            break;  
        }  
    }  
}

前面的都不用看,直接看onClick()方法里面,大家有没有看到,case R.id.login_btn后面的代码有点多,有的人就问啦,不是很多啦,也就十几行。对,也就十几行,但是正式的公司项目中是不可能只有一个点击事件的,也就是switch语句里面会有很多case语句,到时候修改代码的时候可以把你烦死,一个switch块就有几百行几千行代码,会疯掉的~~~


发火

反正我看到这种代码就没好心情了

下面我们稍微改进一下,让代码简洁一些,请看我是如何修改的:

@Override  
public void onClick(View v) {  
    switch (v.getId()) {  
    case R.id.login_btn:  
        //把登录事件封装成了一个方法,直接调用就行了,代码看起来简洁而且容易理解  
        LoginEvent();  
    break;  
    }  
}  

private void LoginEvent() {  
    String username = username_et.getText().toString();  
    String password = password_et.getText().toString();  
    if (username.isEmpty() || password.isEmpty()) {  
        Log.d(TAG, "账号或者密码不能为空");  
    } else if (username.length() < 8 || password.length() < 9) {  
        Log.d(TAG, "账号至少8位,密码至少10位");  
    } else if (username.equals(USERNAME)) {  
        if (password.endsWith(PASSWORD)) {  
            Log.d(TAG, "恭喜你,登录成功");  
        } else {  
            Log.d(TAG, "登录失败->密码错误");  
        }  
    }  
}

其他代码都没有变,我只是把登录事件封装到一个方法体里面了,这样封装一下,会很好理解,代码也就简洁了些。但是现在还有一个问题,如果点击事件很多,你必须要给每个点击事件封装一个方法,夸张点说,如果有100个点击事件,那我岂不是要在MainActivity里面封装100个方法吗?其实挺心疼MainActivity的,一个大姑娘家的,既要处理View层,又要处理业务逻辑层,把管家的事情都做了,能不憔悴吗,能有人喜欢吗

于是MVP架构就出现了,就是为了解决MainActivity大妹子的烦恼,首先我们需要定义一个接口,接口干嘛用的???(握了棵草,也不讲讲清楚,上来就接口!!!)没关系,我在代码中给你注释了,不怕你看不懂

public interface IMainView {  
    //如果登录成功,管家会拨打这个电话(接口)  
    void LoginSuccess();  
    //如果登录失败,管家会拨打这个电话(接口),并告诉MainActivity大妹子为什么登录失败  
    void LoginFailed(String msg);  
}

然后我们让MainActivity实现这个接口

public class MainActivity extends Activity implements OnClickListener, IMainView {  

    private static final String TAG = "MainActivity";  
    public static final String USERNAME = "wuzhanglao";  
    public static final String PASSWORD = "470782682";  

    private EditText username_et;  
    private EditText password_et;  
    private Button login_btn;  
    private MainPresenter mainPresenter;  

    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_main);  
        initView();  
    }  

    private void initView() {  
        username_et = (EditText) findViewById(R.id.username_et);  
        password_et = (EditText) findViewById(R.id.password_et);  
        login_btn = (Button) findViewById(R.id.login_btn);  
        login_btn.setOnClickListener(this);  

        //雇一个叫main的管家,然后把自己登录成功和登录失败的联系方式给他  
        //如果登录成功,管家会拨打loginSuccess这个电话  
        //如果登录失败,管家会拨打logniFailed(String msg)这个电话  
        mainPresenter = new MainPresenter(this);  
    }  

    @Override  
    public void onClick(View v) {  
        switch (v.getId()) {  
        case R.id.login_btn:  
            String username = username_et.getText().toString();  
            String password = password_et.getText().toString();  
            // 让自己的管家去处理登录事件  
            mainPresenter.login(username, password);  
            break;  
        }  
    }  

    @Override  
    public void LoginSuccess() {  
        Log.d(TAG, "管家打电话过来说:登录成功");  
    }  

    @Override  
    public void LoginFailed(String msg) {  
        Log.e(TAG, "管家打电话过来说:登录失败,失败原因:" + msg);  
    }  
}

看到没有,我们的大妹子在onclick点击事件里面让自己的管家来处理登录事件,而自己只需要关心有没有登录成功,这样就显得轻松许多

我们看看大管家是怎么来做事情的

//MainActivity的管家  
public class MainPresenter {  

    private static final String TAG = "MainPresenter";  

    private IMainView view;  

    public MainPresenter(IMainView view) {  
        this.view = view;  
    }  

    // Main管家处理登录逻辑  
    public void login(String username, String password) {  
        if (username.isEmpty() || password.isEmpty()) {  
            view.LoginFailed("账号或密码不能为空");  
        } else if (username.length() < 8 || password.length() < 9) {  
            view.LoginFailed("账号至少8位,密码至少9位");  
        } else if (username.equals(MainActivity.USERNAME)) {  
            if (password.endsWith(MainActivity.PASSWORD)) {  
                // 登录成功,main拨打loginSuccess()电话通知主人登录成功了  
                view.LoginSuccess();  
            } else {  
                // 登录失败,main拨打loginFailed()电话通知主人登录失败了,并告知失败原因  
                view.LoginFailed("密码错误");  
            }  
        }  
    }  
}

大管家会处理登录事件,然后把处理结果告诉大妹子,让大妹子根据不同的结果来进行不同的操作


Paste_Image.png

好了,MVP设计模式就介绍到这里了,如果有看不懂的同学,可以在下方留言,我会给你解答的。看了我的博客,就学会了MVP的同学,可以点个赞啊,收藏下


奋斗

奋斗

最近在写个人微信公众号,有兴趣的可以添加关注一下“代码也是人"


微信公众号二维码.jpeg

文中的错字错句就不要在意啦~~~

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值