`
kevinffk
  • 浏览: 34659 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

[转载]浅谈 MVP in Android

阅读更多

一、概述

对于MVP(Model View Presenter),大多数人都能说出一二:“MVC的演化版本”,“让Model和View完全解耦”等等。本篇博文仅是为了做下记录,提出一些自己的看法,和帮助大家如何针对一个Activity页面去编写针对MVP风格的代码。

对于MVP,我的内心有一个问题:

为何这个模式出来后,就能被广大的Android的程序员接受呢?

 

问了些程序员,他们对于MVP的普遍的认识是:“代码很清晰,不过增加了很多类”。我在第一次看到MVP的时候,看了一个demo,看完以后觉得非常nice(但是回过头来,自己想个例子写,就头疼写不出来,当然这在后文会说)。nice的原因还是因为,这个模式的确让代码的清晰度有了很大的提升。

那么,提升一般都是对比出来的,回顾下,没有应用MVP的代码结构。很多人说明显是MVC么:

  • View:对应于布局文件
  • Model:业务逻辑和实体模型
  • Controllor:对应于Activity

看起来的确像那么回事,但是细细的想想这个View对应于布局文件,其实能做的事情特别少,实际上关于该布局文件中的数据绑定的操作,事件处理的代码都在Activity中,造成了Activity既像View又像Controller(当然了Data-Binder的出现,可能会让View更像View吧)。这可能也就是为何,在该文中有一句这样的话:

Most of the modern Android applications just use View-Model architecture,everything is connected with Activity.

而当将架构改为MVP以后,Presenter的出现,将Actvity视为View层,Presenter负责完成View层与Model层的交互。现在是这样的:

  • View 对应于Activity,负责View的绘制以及与用户交互
  • Model 依然是业务逻辑和实体模型
  • Presenter 负责完成View于Model间的交互

ok,先简单了解下,文中会有例子到时候可以直观的感受下。

小总结下,也就是说,之所以让人觉得耳目一新,是因为这次的跳跃是从并不标准的MVCMVP的一个转变,减少了Activity的职责,简化了Activity中的代码,将复杂的逻辑代码提取到了Presenter中进行处理。与之对应的好处就是,耦合度更低,更方便的进行测试。借用两张图(出自:该文),代表上述的转变:

转变为:

二、MVP 与 MVC 区别

ok,上面说了一堆理论,下面我们还是需要看一看MVC与MVP的一个区别,请看下图(来自:本文):

其实最明显的区别就是,MVC中是允许Model和View进行交互的,而MVP中很明显,Model与View之间的交互由Presenter完成。还有一点就是Presenter与View之间的交互是通过接口的(代码中会体现)。

还有一堆概念性的东西,以及优点就略了,有兴趣自行百度。下面还是通过一些简单的需求来展示如何编写MVP的demo。

三、Simple Login Demo

效果图:

看到这样的效果,先看下完工后的项目结构:

ok,接下来开始一步一步的编写思路。

(一)Model

首先实体类User不用考虑这个肯定有,其次从效果图可以看到至少有一个业务方法login(),这两点没什么难度,我们首先完成:

package com.zhy.blogcodes.mvp.bean;

/**
 * Created by zhy on 15/6/18.
 */
public class User
{
    private String username ;
    private String password ;

    public String getUsername()
    {
        return username;
    }

    public void setUsername(String username)
    {
        this.username = username;
    }

    public String getPassword()
    {
        return password;
    }

    public void setPassword(String password)
    {
        this.password = password;
    }
}

 

package com.zhy.blogcodes.mvp.biz;

/**
 * Created by zhy on 15/6/19.
 */
public interface IUserBiz
{
    public void login(String username, String password, OnLoginListener loginListener);
}

 

package com.zhy.blogcodes.mvp.biz;

import com.zhy.blogcodes.mvp.bean.User;

/**
 * Created by zhy on 15/6/19.
 */
public interface OnLoginListener
{
    void loginSuccess(User user);

    void loginFailed();
}

 

实体类不用说,至于业务类,我们抽取了一个接口,一个实现类这也很常见~~login方法,一般肯定是连接服务器的,是个耗时操作,所以我们开辟了子线程,Thread.sleep(2000)模拟了耗时,由于是耗时操作,所以我们通过一个回调接口来通知登录的状态。

其实这里还是比较好写的,因为和传统写法没区别。

(二) View

上面我们说过,Presenter与View交互是通过接口。所以我们这里需要定义一个ILoginView,难点就在于应该有哪些方法,我们看一眼效果图:

可以看到我们有两个按钮,一个是login,一个是clear;

login说明了要有用户名、密码,那么对应两个方法:

String getUserName();

String getPassword();

 

再者login是个耗时操作,我们需要给用户一个友好的提示,一般就是操作ProgressBar,所以再两个:
void showLoading();

void hideLoading();
 login当然存在登录成功与失败的处理,我们主要看成功我们是跳转Activity,而失败可能是去给个提醒:
void toMainActivity(User user);

void showFailedError()
 ok,login这个方法我们分析完了~~还剩个clear那就简单了:
    void clearUserName();

    void clearPassword();
 综上,接口完整为:
package com.zhy.blogcodes.mvp.view;

import com.zhy.blogcodes.mvp.bean.User;

/**
 * Created by zhy on 15/6/19.
 */
public interface IUserLoginView
{
    String getUserName();

    String getPassword();

    void clearUserName();

    void clearPassword();

    void showLoading();

    void hideLoading();

    void toMainActivity(User user);

    void showFailedError();

}
 

有了接口,实现就太好写了~~~

总结下,对于View的接口,去观察功能上的操作,然后考虑:

  • 该操作需要什么?(getUserName, getPassword)
  • 该操作的结果,对应的反馈?(toMainActivity, showFailedError)
  • 该操作过程中对应的友好的交互?(showLoading, hideLoading)

下面贴一下我们的View的实现类,哈,其实就是Activity,文章开始就说过,MVP中的View其实就是Activity。

package com.zhy.blogcodes.mvp;

import android.os.Bundle;
import android.support.v7.app.ActionBarActivity;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.Toast;

import com.zhy.blogcodes.R;
import com.zhy.blogcodes.mvp.bean.User;
import com.zhy.blogcodes.mvp.presenter.UserLoginPresenter;
import com.zhy.blogcodes.mvp.view.IUserLoginView;

public class UserLoginActivity extends ActionBarActivity implements IUserLoginView
{


    private EditText mEtUsername, mEtPassword;
    private Button mBtnLogin, mBtnClear;
    private ProgressBar mPbLoading;

    private UserLoginPresenter mUserLoginPresenter = new UserLoginPresenter(this);

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

        initViews();
    }

    private void initViews()
    {
        mEtUsername = (EditText) findViewById(R.id.id_et_username);
        mEtPassword = (EditText) findViewById(R.id.id_et_password);

        mBtnClear = (Button) findViewById(R.id.id_btn_clear);
        mBtnLogin = (Button) findViewById(R.id.id_btn_login);

        mPbLoading = (ProgressBar) findViewById(R.id.id_pb_loading);

        mBtnLogin.setOnClickListener(new View.OnClickListener()
        {
            @Override
            public void onClick(View v)
            {
                mUserLoginPresenter.login();
            }
        });

        mBtnClear.setOnClickListener(new View.OnClickListener()
        {
            @Override
            public void onClick(View v)
            {
                mUserLoginPresenter.clear();
            }
        });
    }


    @Override
    public String getUserName()
    {
        return mEtUsername.getText().toString();
    }

    @Override
    public String getPassword()
    {
        return mEtPassword.getText().toString();
    }

    @Override
    public void clearUserName()
    {
        mEtUsername.setText("");
    }

    @Override
    public void clearPassword()
    {
        mEtPassword.setText("");
    }

    @Override
    public void showLoading()
    {
        mPbLoading.setVisibility(View.VISIBLE);
    }

    @Override
    public void hideLoading()
    {
        mPbLoading.setVisibility(View.GONE);
    }

    @Override
    public void toMainActivity(User user)
    {
        Toast.makeText(this, user.getUsername() +
                " login success , to MainActivity", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void showFailedError()
    {
        Toast.makeText(this,
                "login failed", Toast.LENGTH_SHORT).show();
    }
}

 

对于在Activity中实现我们上述定义的接口,是一件很容易的事,毕竟接口引导我们去完成。

最后看我们的Presenter。

(三)Presenter

Presenter是用作Model和View之间交互的桥梁,那么应该有什么方法呢?

其实也是主要看该功能有什么操作,比如本例,两个操作:login和clear。

package com.zhy.blogcodes.mvp.presenter;

import android.os.Handler;

import com.zhy.blogcodes.mvp.bean.User;
import com.zhy.blogcodes.mvp.biz.IUserBiz;
import com.zhy.blogcodes.mvp.biz.OnLoginListener;
import com.zhy.blogcodes.mvp.biz.UserBiz;
import com.zhy.blogcodes.mvp.view.IUserLoginView;


/**
 * Created by zhy on 15/6/19.
 */
public class UserLoginPresenter
{
    private IUserBiz userBiz;
    private IUserLoginView userLoginView;
    private Handler mHandler = new Handler();

    public UserLoginPresenter(IUserLoginView userLoginView)
    {
        this.userLoginView = userLoginView;
        this.userBiz = new UserBiz();
    }

    public void login()
    {
        userLoginView.showLoading();
        userBiz.login(userLoginView.getUserName(), userLoginView.getPassword(), new OnLoginListener()
        {
            @Override
            public void loginSuccess(final User user)
            {
                //需要在UI线程执行
                mHandler.post(new Runnable()
                {
                    @Override
                    public void run()
                    {
                        userLoginView.toMainActivity(user);
                        userLoginView.hideLoading();
                    }
                });

            }

            @Override
            public void loginFailed()
            {
                //需要在UI线程执行
                mHandler.post(new Runnable()
                {
                    @Override
                    public void run()
                    {
                        userLoginView.showFailedError();
                        userLoginView.hideLoading();
                    }
                });

            }
        });
    }

    public void clear()
    {
        userLoginView.clearUserName();
        userLoginView.clearPassword();
    }



}

 

注意上述代码,我们的presenter完成二者的交互,那么肯定需要二者的实现类。大致就是从View中获取需要的参数,交给Model去执行业务方法,执行的过程中需要的反馈,以及结果,再让View进行做对应的显示。

ok,拿到一个例子经过上述的分解基本就能完成。以上纯属个人见解,欢迎讨论和吐槽~ have a nice day ~。

参考资料

分享到:
评论

相关推荐

    浅谈Android MVP

    **Android MVP 模式详解** Android MVP(Model-View-Presenter)模式是一种常见的软件设计模式,用于改善Android应用的架构,提高代码的可测试性和可维护性。在Android开发中,MVP帮助开发者分离界面展示逻辑与业务...

    浅谈MVP.doc

    ### 浅谈MVP与Model-View-ViewModel(MVVM)设计模式 #### 一、MVP设计模式概述 在软件工程中,特别是在GUI应用程序开发领域,为了更好地组织代码结构和提高开发效率,设计师和开发者们发展出了多种设计模式。其中...

    Android MVPdemo

    **Android MVP 模式详解** 在移动应用开发中,Model-View-Presenter(MVP)设计模式是一种常用架构,尤其在Android平台上。本项目“Android MVPdemo”提供了一个基础的MVP实现,包括登录界面、首页展示以及数据加载...

    Android代码-Android_Blog_Demos

    浅谈 MVP in Android Android ViewDragHelper完全解析 自定义ViewGroup神器 Android HandlerThread 完全解析 Android IntentService完全解析 当Service遇到Handler 详细 Android 高清加载巨图方案 拒绝压缩图片 ...

    android ioc与mvp

    在Android开发中,IOC(Inversion of Control,控制反转)和MVP(Model-View-Presenter,模型-视图- presenter)是两种重要的设计模式,它们有助于提高代码的可维护性和可测试性。本Demo旨在结合这两者,提供一个...

    android MVP实践

    在Android应用开发中,Model-View-Presenter(MVP)是一种常见的设计模式,它将应用程序的业务逻辑、用户界面和数据处理分离,提高了代码的可测试性和可维护性。MVP模式是为了解决传统的Model-View模式在复杂项目中...

    Android使用mvp模式入门

    **Android MVP模式详解** 在Android应用开发中,MVP(Model-View-Presenter)模式是一种常用的架构设计模式,它有助于实现清晰的职责分离,提高代码的可测试性和可维护性。MVP模式由三个主要组件构成:Model(模型...

    Android mvp

    Android MVP(Model-View-Presenter)是一种设计模式,常用于Android应用开发中,以提高代码的可维护性和可测试性。MVP模式是基于经典的MVC(Model-View-Controller)模式演变而来的,它将业务逻辑、用户界面和数据...

    Android MVP模式示例

    在Android开发中,MVP(Model-View-Presenter)模式是一种常见的设计模式,它将业务逻辑、用户界面和数据模型分离,使得代码更加结构化、可测试和易于维护。本示例将深入探讨如何在Android项目中应用MVP模式。 **一...

    android mvp_app

    在Android应用开发中,Model-View-Presenter(MVP)是一种流行的设计模式,它有助于实现清晰的代码结构和良好的解耦。"android mvp_app"是基于这种模式的一个示例应用,旨在帮助开发者理解并实践MVP。在这个项目中,...

    MVP_Android_Demo

    **Android开发中的MVP模式详解** MVP(Model-View-Presenter)模式是软件设计模式中的一种,尤其在Android开发中被广泛采用。MVP模式的主要目的是为了分离业务逻辑、界面展示和数据处理,提高代码的可测试性和可...

    architecture-samples-todo-mvp_Android官方MVP架构示例_android_balllyd_源

    **Android MVP架构详解** 在移动应用开发中,保持代码的清晰性和可维护性至关重要。MVP(Model-View-Presenter)架构模式就是一种常见的设计模式,它有助于将业务逻辑、用户界面和数据模型分离,从而提高代码的组织...

    基于mvp框架android的代码(as3.5)

    【Android MVP框架详解】 在Android应用开发中,Model-View-Presenter(MVP)是一种常见的设计模式,它有助于实现清晰的代码结构,提高代码可测试性,并降低耦合度。这个"基于mvp框架android的代码(as3.5)"项目...

    android mvp代码生成插件

    mvp模板代码自动生成,简化大家采用mvp开发时需要写大量重复模板代码的工作。大家下载好后,androidStudio选择从本地安装插件,选择你下载插件存放的文件夹,安装,重启androidstudio即可使用

    AndroidMVP.zip

    在Android应用开发中,MVP(Model-View-Presenter)是一种常见的设计模式,它强调了组件之间的职责分离,提高了代码的可测试性和可维护性。本项目"AndroidMVP.zip"是一个采用RxJava、Retrofit以及MVP架构的示例项目...

    android MVP架构demo

    在Android应用开发中,MVP(Model-View-Presenter)架构是一种常见的设计模式,它将应用程序的业务逻辑、用户界面和数据模型分离,以提高代码的可测试性、可维护性和可扩展性。本Android MVP架构demo展示了如何在...

    android mvp 模拟登陆

    在Android应用开发中,Model-View-Presenter(MVP)是一种常见的设计模式,它有助于分离界面展示、数据处理和业务逻辑,从而提高代码的可测试性和可维护性。本示例"android mvp 模拟登陆"是一个使用MVP模式实现的...

    android mvp模式demo

    **Android MVP模式详解** 在Android应用开发中,Model-View-Presenter(MVP)模式是一种常用的架构设计模式,它有助于实现清晰的代码组织,提高代码的可测试性和可维护性。本项目“android mvp模式demo”提供了一个...

    android mvp

    android mvpandroid mvpandroid mvpandroid mvpandroid mvpandroid mvpandroid mvpandroid mvpandroid mvpandroid mvp

    android mvp架构示例

    Android MVP架构示例详解 MVP(Model-View-Presenter)是一种常见的软件设计模式,尤其在Android开发中被广泛采用,以实现清晰的代码结构和易于测试的特性。MVP模式将应用程序分为三个主要组件:Model(模型)、...

Global site tag (gtag.js) - Google Analytics