详解Android中的MVP架构分解和实现

1、概述

传统的Android开发架构一般是MVC模式,

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

单独从逻辑看起来非常好,与我们做Web开发时,开发模式类似,但在实际开发中,View对应于布局文件,实际上关于该布局文件中的数据绑定的操作,事件处理的代码都在Activity中,Activity既像View又像Controller(MVVP架构中包括数据绑定),导致Activity中职责太重,耦合度大。修改和维护起来非常麻烦。

2、MVP介绍

MVP架构中,View 对应于Activity,负责View的绘制以及与用户交互

Model 依然是业务逻辑和实体模型,Presenter 负责完成View于Model间的交互。

(1)Model层

模型层之中做的工作是具体业务逻辑处理的实现,都伴随着程序中各种数据的处理,复杂一些的就需要实现一个Interface来松耦合了。

(2)View层

视图层体现的很轻薄,负责显示数据、提供友好界面跟用户交互就行。MVP下Activity和Fragment体现在了这一 层,Activity一般也就做加载UI视图、设置监听再交由Presenter处理的一些工作,所以也就需要持有相应Presenter的引用。处理一些基本UI逻辑,判断是否为空。

(3)Presenter层

Presenter这一层处理着程序各种逻辑的分发,收到View层UI上的反馈命令、定时命令、系统命令等指令后分发处理逻辑交由Model层做具体的业务操作。

MVP架构和MVC架构区别:

MVC中是允许Model和View进行交互的,而MVP中,Model与View之间的交互由Presenter完成。还有一点就是Presenter与View之间的交互是通过接口的。

3、MVP实现

百说不如一做。实现一个简单的登录操作。

项目结构如下:

(1)Model层实现

首先实现User实体类:

package com.chunsoft.blogcontent.bean;

/**
 * Developer:chunsoft on 2017/2/7 11:19
 * Content:实体类
 */

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;
 }
}

model层主要实现业务逻辑处理,在本文案例中,主要逻辑处理就是登录,抽取了一个接口和一个实现类,在login操作,模拟登录操作,Thread.sleep()模拟耗时,由于是耗时操作,通过一个回调接口通知登录状态。

model层接口:

package com.chunsoft.blogcontent.model;

import com.chunsoft.blogcontent.RequestCallback;
import com.chunsoft.blogcontent.bean.User;

/**
 * Developer:chunsoft on 2017/2/7 11:30
 * Email:chun_soft@qq.com
 * Content:Model层主要处理业务方法和实体模型
 */

public interface LoginModel {
 void login(String username, String password, RequestCallback<User> callback);
}

model层接口实现:

package com.chunsoft.blogcontent.model.impl;

import com.chunsoft.blogcontent.RequestCallback;
import com.chunsoft.blogcontent.bean.User;
import com.chunsoft.blogcontent.model.LoginModel;

/**
 * Developer:chunsoft on 2017/2/7 11:31
 * Content:Model层的实现
 */

public class LoginModelImpl implements LoginModel{
 @Override
 public void login(final String username, final String password, final RequestCallback<User> callback) {
  //模仿登录操作
  new Thread() {
   @Override
   public void run() {
    try {
     Thread.sleep(2*1000);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    if ("chunsoft".equals(username) && "123456".equals(password)) {
     User user = new User();
     user.setUsername(username);
     user.setPassword(password);
     callback.onSuccess(user);
    } else {
     callback.onFailure("登录失败");
    }
   }
  }.start();
 }
}

回调接口:

package com.chunsoft.blogcontent;

/**
 * Developer:chunsoft on 2017/2/7 11:35
 * Content:回调接口
 */

public interface RequestCallback<T> {
 void onSuccess(T datas);
 void onFailure(String msg);
}

(2)View层实现

对于View层接口定义,首先考虑功能上的操作,然后考虑:

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

view层接口:

package com.chunsoft.blogcontent.view;

import com.chunsoft.blogcontent.bean.User;

/**
 * Developer:chunsoft on 2017/2/7 11:54
 * Email:chun_soft@qq.com
 * Content:view层接口
 */

public interface LoginView {
 String getUserName();
 String getPassword();

 void showLoading();
 void hideLoading();

 void toMainActvity(User user);
 void showFailedError(String msg);
}

view层实现,其实就是Activity,可以看到Activity的代码量大大减少,且逻辑清晰:

package com.chunsoft.blogcontent;

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

import com.chunsoft.blogcontent.bean.User;
import com.chunsoft.blogcontent.presenter.impl.LoginPresenterImpl;
import com.chunsoft.blogcontent.view.LoginView;

public class MainActivity extends AppCompatActivity implements LoginView{

 private EditText et_mobile,et_password;
 private ProgressBar pb;
 private Button btn_login;

 private LoginPresenterImpl mLoginPresenter = new LoginPresenterImpl(this);

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

  et_mobile = (EditText) findViewById(R.id.et_mobile);
  et_password = (EditText) findViewById(R.id.et_password);
  pb = (ProgressBar) findViewById(R.id.pb);
  btn_login = (Button) findViewById(R.id.btn_login);

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

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

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

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

 @Override
 public void hideLoading() {
  pb.setVisibility(View.INVISIBLE);
 }

 @Override
 public void toMainActvity(User user) {
  Toast.makeText(this,user.getUsername() + "登录成功",Toast.LENGTH_SHORT).show();
 }

 @Override
 public void showFailedError(String msg) {
  Toast.makeText(this,msg,Toast.LENGTH_SHORT).show();
 }
}

(3)Presenter层实现

Presenter层是Model层和View层的桥梁,本文中依然抽象出一个接口和一个实现类,定义时主要看该功能有什么操作,例如,登录:

Presenter接口:

package com.chunsoft.blogcontent.presenter;

/**
 * Developer:chunsoft on 2017/2/7 11:49
 * Content:Presenter层接口
 */
public interface LoginPresenter {
  void login();
}

Presenter层实现:

package com.chunsoft.blogcontent.presenter.impl;

import android.os.Handler;

import com.chunsoft.blogcontent.RequestCallback;
import com.chunsoft.blogcontent.bean.User;
import com.chunsoft.blogcontent.model.LoginModel;
import com.chunsoft.blogcontent.model.impl.LoginModelImpl;
import com.chunsoft.blogcontent.presenter.LoginPresenter;
import com.chunsoft.blogcontent.view.LoginView;


/**
 * Developer:chunsoft on 2017/2/7 11:50
 * Content:Presenter层实现
 */

public class LoginPresenterImpl implements LoginPresenter {

 private LoginView loginView;
 private LoginModel loginModel;

 private Handler mHandler = new Handler();

 //在构造函数中初始化
 public LoginPresenterImpl(LoginView loginView) {
  this.loginModel = new LoginModelImpl();
  this.loginView = loginView;
 }

 @Override
 public void login() {
  loginView.showLoading();
  loginModel.login(loginView.getUserName(), loginView.getPassword(), new RequestCallback<User>() {
   @Override
   public void onSuccess(final User datas) {
    //登录成功
    mHandler.post(new Runnable() {
     @Override
     public void run() {
      loginView.toMainActvity(datas);
      loginView.hideLoading();
     }
    });
   }

   @Override
   public void onFailure(final String msg) {
    //登录失败
    mHandler.post(new Runnable() {
     @Override
     public void run() {
      loginView.showFailedError(msg);
      loginView.hideLoading();
     }
    });
   }
  });
 }
}

Presenter层是Model层和View层的桥梁,Model层和View层不直接通信,所以Presenter层需要Model层和View层的实现类,从View层中获取重要参数,交给Model 层调用业务逻辑处理,执行后的结果和反馈再交给View层去展示。

本文源码下载

本文是简单的用MVP架构实现登录操作,而目前主流开发,将Rxjava、Retrofit和MVP结合进行开发,逻辑更加清晰,下篇文章将详细介绍这部分实现。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持呐喊教程。

声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:notice#nhooo.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。