解析Android的MVC和MVP模式

1.     MVC

         mvc是model、view、controller的缩写。Android 鼓励弱耦合和组件的重用,android 中mvc的具体体现如下:

           · 模型(model):用于封装与应用程序的业务逻辑相关的数据以及对数据的处理方法。“模型”有对数据直接访问的权力,例如对数据库的访问。模型中数据的变化一般会通过一种刷新机制被公布。为了实现这种机制,那些用于监视此模型的视图必须事先在此模型上注册,从而,视图可以了解数据模型发生的改变。(比较:观察者模式软件设计模式))(对数据库的操作、对网络等的操作都应该在model里面处理,当然对计算等操作也是必须放在该层的)。

           · 视图层(view):是应用程序中负责生成用户界面的部分。也是整个mvc架构中用户唯一可以看到的一层,接收用户的输入,显示用户的处理结果。为了实现视图上的刷新功能,视图需要访问它监视的数据模型(Model),因此应该事先在被它监视的数据那里注册。

          ·控制层(controller):根据用户的输入,控制用户界面数据显示及更新model对象状态的部分。onKeyUponKeyMultiple,etc.这些输入都可能改变业务逻辑数据而引起view的变化。也就是说Event(事件)导致Controller改变了Model或View,或者同时改变两者。只要Controller改变了Model的数据,所有依赖该数据的View都要更新。似的,只要Controller改变了View,View会从潜在的Model中获取数据(没必要的情况下可能不必获取)来刷新自己。

2.     MVP

         MVP是MVC模式的演化版本,本篇文章对MVP的介绍目的是帮助大家如何针对一个Activity页面去编写出MVP风格的代码。

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

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

·View:对应于布局文件

·Model:业务逻辑和实体模型

·Controllor:对应于Activity

         看起来的确像那么回事,但是细细的想想这个View对应于布局文件,其实能做的事情特别少,实际上关于该布局文件中的数据绑定的操作,事件处理的代码都在Activity中,造成了Activity既像View又像Controller。

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

·View 对应于Activity,负责View的绘制以及与用户交互

·Model 依然是业务逻辑和实体模型

·Presenter 负责完成View于Model间的交互

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

         MVCMVP最明显的区别就是,MVC中是允许ModelView进行交互的,而MVP中很明显,ModelView之间的交互由Presenter完成。还有一点就是PresenterView之间的交互是通过接口的。

下面通过一些简单的例子来展示如何编写MVPdemo

效果图:


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


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(Stringusername)

    {

       this.username = username;

    }

 

    public String getPassword()

    {

       return password;

    }

 

    public void setPassword(Stringpassword)

    {

       this.password = password;

    }

}

package com.zhy.blogcodes.mvp.biz;

 

/**

 * Created by zhy on 15/6/19.

 */

public interface IUserBiz

{

    public void login(Stringusername, 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 class UserBizimplements IUserBiz

{

 

    @Override

    public void login(final String username, finalString password, final OnLoginListenerloginListener)

    {

       //模拟子线程耗时操作

       new Thread()

       {

           @Override

           public voidrun()

           {

                try

                {

                    Thread.sleep(2000);

                } catch(InterruptedException e)

                {

                   e.printStackTrace();

                }

                //模拟登录成功

                if("zhy".equals(username) &&"123".equals(password))

                {

                    User user = new User();

                    user.setUsername(username);

                    user.setPassword(password);

                   loginListener.loginSuccess(user);

                } else

                {

                   loginListener.loginFailed();

                }

           }

       }.start();

    }

}

 

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 UserLoginActivityextends ActionBarActivityimplements IUserLoginView

{

 

 

    private EditText mEtUsername, mEtPassword;

    private Button mBtnLogin, mBtnClear;

    private ProgressBar mPbLoading;

 

    private UserLoginPresenter mUserLoginPresenter = new UserLoginPresenter(this);

 

    @Override

    protected void onCreate(BundlesavedInstanceState)

    {

       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(newView.OnClickListener()

       {

           @Override

           public voidonClick(View v)

           {

                mUserLoginPresenter.login();

           }

       });

 

       mBtnClear.setOnClickListener(newView.OnClickListener()

       {

           @Override

           public voidonClick(View v)

           {

                mUserLoginPresenter.clear();

           }

       });

    }

 

 

    @Override

    public String getUserName()

    {

       returnmEtUsername.getText().toString();

    }

 

    @Override

    public String getPassword()

    {

       returnmEtPassword.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(Useruser)

    {

       Toast.makeText(this,user.getUsername() +

                "login success , to MainActivity", Toast.LENGTH_SHORT).show();

    }

 

    @Override

    public void showFailedError()

    {

       Toast.makeText(this,

                "loginfailed", Toast.LENGTH_SHORT).show();

    }

}

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

最后看我们的Presenter。

(三)Presenter

        Presenter是用作Model和View之间交互的桥梁,那么应该有什么方法呢?其实也是主要看该功能有什么操作,比如本例,两个操作:loginclear

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(IUserLoginViewuserLoginView)

    {

       this.userLoginView = userLoginView;

       this.userBiz = new UserBiz();

    }

 

    public void login()

    {

       userLoginView.showLoading();

       userBiz.login(userLoginView.getUserName(), userLoginView.getPassword(), new OnLoginListener()

       {

           @Override

           public voidloginSuccess(final User user)

           {

                //需要在UI线程执行

                mHandler.post(new Runnable()

                {

                    @Override

                    publicvoid run()

                    {

                       userLoginView.toMainActivity(user);

                       userLoginView.hideLoading();

                    }

                });

 

           }

 

           @Override

           public voidloginFailed()

           {

                //需要在UI线程执行

                mHandler.post(new Runnable()

                {

                    @Override

                    publicvoid run()

                    {

                       userLoginView.showFailedError();

                       userLoginView.hideLoading();

                    }

                });

 

           }

       });

    }

 

    public void clear()

    {

       userLoginView.clearUserName();

       userLoginView.clearPassword();

    }

}

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

         参考http://blog.csdn.net/lmj623565791/article/details/46596109

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值