0. 前言

为了更好地进行移动端架构设计,我们最常用的就是MVC、MVP和MVVM,作为三个最耳熟能详的三大架构,应用可谓非常广泛。对于这三种架构设计以及优缺点已经在Android APP架构设计——MVC、MVP和MVVM介绍一文中介绍过了,本文是对前面那篇文章2.3小节的补充,介绍MVP模式在Android中的使用示例,目的在于深化对MVP架构的理解。

1.   使用场景

这里我们实现一个简单的登录功能。先看一下效果图。


1.1   Model层设计

Model层包括我们的基本实体类User,维护用户名和用户密码。

/**
* User Bean Class
* Created by SEU_Calvin on 2016/10/25.
*/
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;
}
}

接下来是登录接口和它的实现类,参数中用户名和密码没什么好说的,第三个参数是我们设置的一个回调接口,来通知我们登录的状态,即成功or失败。最后我们在登录的实现类中通过子线程模拟真实的登录耗时任务,在登录成功or失败后回调接口中的loginSuccess or loginFailed。

/**
* Login Interface with its Implement. And the LoginListener.
* Created by SEU_Calvin on 2016/10/25.
*/
public interface ILogin {
void login(String username, String password, OnLoginListener loginListener);
}
public interface OnLoginListener {
void loginSuccess(User user);
void loginFailed();
}
public class ILoginImpl implements ILogin {
@Override
public void login(final String username, final String password, final OnLoginListener loginListener) {
//模拟子线程耗时操作
new Thread() {
@Override
public void run() {
try {
Thread.sleep(1500);
} catch (InterruptedException e) {
e.printStackTrace();
}
//登录成功
if ("SEU".equals(username) && "123456".equals(password)) {
User user = new User();
user.setUsername(username);
user.setPassword(password);
loginListener.loginSuccess(user);
} else {
//登录失败
loginListener.loginFailed();
}
}
}.start();
}
}


1.2  View层

前面介绍中已经提到Presenter与View交互是通过接口,本例中充当该角色的是我们的IUserLogin。最后让我们的Activity实现这个接口。

/**
* The Interface Activity should Implement.
* Created by SEU_Calvin on 2016/10/25.
*/
public interface IUserLogin {
String getUserName();
String getPassword();
void clearUserName();
void clearPassword();
void showLoading();
void hideLoading();
void showLoginSuccess(User user);
void showLoginFail();
}
public class MainActivity extends AppCompatActivity implements IUserLogin{
private EditText et_userPw, et_userName;
private Button login, clear;
private ProgressBar progressBar;
//持有Presenter的引用
private UserLoginPresenter mUserLoginPresenter = new UserLoginPresenter(this);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initViews();
} private void initViews(){
et_userName = (EditText) findViewById(R.id.et_userName);
et_userPw = (EditText) findViewById(R.id.et_userPw);
clear = (Button) findViewById(R.id.clear);
login = (Button) findViewById(R.id.login);
progressBar = (ProgressBar) findViewById(R.id.progressBar);
login.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//登录操作还是交给了Presenter去控制
mUserLoginPresenter.login();
}
}); clear.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mUserLoginPresenter.clear();
}
});
} @Override
public String getUserName() {
return et_userName.getText().toString();
} @Override
public String getPassword() {
return et_userPw.getText().toString();
} @Override
public void clearUserName() {
et_userName.setText("");
} @Override
public void clearPassword() {
et_userPw.setText("");
} @Override
public void showLoading() {
progressBar.setVisibility(View.VISIBLE);
} @Override
public void hideLoading() {
progressBar.setVisibility(View.GONE);
} @Override
public void showLoginSuccess(User user) {
Toast.makeText(this, user.getUsername() + " login success", Toast.LENGTH_SHORT).show();
} @Override
public void showLoginFail() {
Toast.makeText(this, "login fail", Toast.LENGTH_SHORT).show();
}
}

看IUserLogin接口中的方法就知道,里面的很多功能如果是传统MVC写法,是全部写在Activity里的,拿ProgressBar来做例子,传统的写法会有ProgressBar什么时候显示什么时候隐藏的逻辑判断过程,这显然是Model层的逻辑,在MVP里面实现了和View层的解耦。真正实现Model层和View层交互的是我们的Presenter层。

1.3  Presenter层

/**
* Presenter.
* Created by SEU_Calvin on 2016/10/25.
*/
public class UserLoginPresenter{
private ILogin login;
private IUserLogin userLogin;
private Handler mHandler = new Handler(); public UserLoginPresenter(IUserLogin userLoginView) {
//这里传入对Activity的引用,即View层的引用
this.userLogin = userLoginView;
//这里是对Model层的引用
this.login = new ILoginImpl();
} public void login() {
userLogin.showLoading();
login.login(userLogin.getUserName(), userLogin.getPassword(), new OnLoginListener() {
@Override
public void loginSuccess(final User user) {
//需要在UI线程执行
mHandler.post(new Runnable() {
@Override
public void run() {
userLogin.showLoginSuccess(user);
userLogin.hideLoading();
}
});
}
@Override
public void loginFailed() {
//需要在UI线程执行
mHandler.post(new Runnable() {
@Override
public void run() {
userLogin.showLoginFail();
userLogin.hideLoading();
}
});
}
});
}
public void clear() {
userLogin.clearUserName();
userLogin.clearPassword();
}
}

从Presenter层的表现来看,它把作为Model层和View层中间人的作用发挥的淋漓尽致。Presenter同时持有Activity(View层)和ILoginImpl(Model层)的引用,先从View中获取需要的参数,再交给Model去执行业务方法,执行的结果通过接口的方式通过Presenter层传递给View层,最后进行显示。

2.   内存泄漏的问题

由于Presenter 经常性的持有Activity 的强引用,如果在一些请求结束之前Activity 被销毁了,Activity对象将无法被回收,此时就会发生内存泄露。这里我们使用虚引用和泛型来对MVP中的内存泄漏问题进行改良。

2.1   问题解决

(1)首先Model层是不用修改的。

(2)其次View层需要抽象出一层父类BaseActivity,并利用Activity的生命周期方法对View层和Presenter层实现绑定和解绑。

/**
* The improvement of View, Abstract BaseClass form MainActivity.
* Created by SEU_Calvin on 2016/10/25.
*/
public abstract class BaseActivity <V,T extends BasePresenter<V>> extends AppCompatActivity {
protected T mUserLoginPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mUserLoginPresenter = createPresenter();
mUserLoginPresenter.attachView((V) this);
}
@Override
protected void onDestroy() {
super.onDestroy();
mUserLoginPresenter.detachView();
}
protected abstract T createPresenter();
}
public class MainActivity extends BaseActivity< IUserLogin,LoginPresenter> implements IUserLogin{
private UserLoginPresenter presenter;
@Override
protected UserLoginPresenter createPresenter() {
presenter = new UserLoginPresenter (this);
return presenter;
}
//initView方法逻辑不变,IUserLogin接口实现逻辑不变
}

(3)最后Presenter层也抽象出BasePresenter类,使Presenter层持有Activity的软引用。

/**
* The improvement of Presenter, Abstract BaseClass form UserLoginPresenter.
* Created by SEU_Calvin on 2016/10/25.
*/
public abstract class BasePresenter<T> {
protected Reference<T> viewRef;
public void attachView(T view){
//持有的是Activity的软引用,
viewRef= new WeakReference<T>(view);
}
public void detachView(){
if(viewRef !=null){
viewRef.clear();
viewRef=null;
}
}
}
public class UserLoginPresenter extends BasePresenter< IUserLogin>{
//其余逻辑不变
}

以上便是MVP在Android中的一个简单实现示例,并对其中暴露出的内存泄露问题提出的优化方案。

希望对你有所帮助,请大家多点赞支持。

Android APP架构设计——MVP的使用示例的更多相关文章

  1. Android APP架构设计——MVC、MVP和MVVM介绍

    )对于过大的项目,数据绑定需要花费更多的内存. 关于APP的架构设计就介绍到这吧,转载请注明出处:http://blog.csdn.net/seu_calvin/article/details/529 ...

  2. android app 架构设计02

    二:在开放的过程中,尽量把工具类,BaseActivity 放在指定的位置. DateFormat Bitmap Notification Shared Preference Environment ...

  3. android app 架构设计01

    1:本文有摘抄, 1 2 3 4 5 - 开发过程中.需求.设计.编码的一致性 - 整个程序具有统一的风格,比方对话框样式,button风格,色调等UI元素 - 整个程序详细统一的结构,比方不同模块訪 ...

  4. Android App的设计架构:MVC,MVP,MVVM与架构经验谈

    相关:http://www.cnblogs.com/wytiger/p/5996876.html 和MVC框架模式一样,Model模型处理数据代码不变在Android的App开发中,很多人经常会头疼于 ...

  5. Android App的设计架构:MVC,MVP,MVVM与架构AAAAA

    1. 架构设计的目的1.1 通过设计使程序模块化,做到模块内部的高聚合和模块之间的低耦合.1.2 这样做的好处是使得程序在开发的过程中,开发人员只需要专注于一点,提高程序开发的效率,并且更容易进行后续 ...

  6. 【转】App架构设计经验谈:接口的设计

    App架构设计经验谈:接口的设计 App与服务器的通信接口如何设计得好,需要考虑的地方挺多的,在此根据我的一些经验做一些总结分享,旨在抛砖引玉. 安全机制的设计 现在,大部分App的接口都采用REST ...

  7. Android App组件之ListFragment -- 说明和示例

    Android App组件之ListFragment -- 说明和示例 1 ListFragement介绍 ListFragment继承于Fragment.因此它具有Fragment的特性,能够作为a ...

  8. Hybrid APP 架构设计思路

    关于Hybrid模式开发app的好处,网络上已有很多文章阐述了,这里不展开. 本文将从以下几个方面阐述Hybrid app架构设计的一些经验和思考. 原文及讨论请到 github issue 通讯 作 ...

  9. Android App组件之Fragment说明和示例

    Android App组件之Fragment说明和示例 1 Fragement介绍 Android从3.0开始引入Fragment,主要是为了支持更动态更灵活的界面设计. Fragment是activ ...

随机推荐

  1. Sql去重一些技巧

    下午的时候遇到点问题,Sql去重,简单的去重可以用 DISTINCT 关键字去重,不过,很多情况下用这个解决不了问题.重复的数据千变万化,例如:类似于qq.微信的最近联系人功能,读取这些数据肯定要和消 ...

  2. 在Go语言中记录log:seelog包

    前两周调bug调的吐血,虽然解决了但是还是挺浪费时间的.跟同事聊了聊,觉得我们现在项目中的日志记录太少了,导致出了问题不知道怎么下手,还得自己改代码记录日志,然后排查问题.这样如果将来还有bug的话还 ...

  3. php仿照asp实现application缓存的代码分享

    php 怎么没有asp 那样的application缓存呢?最近我找了很多,都只有自己写,下面我分享一段代码 class php_cache{ //相对或者绝对目录,末尾不要加 '/' var $ca ...

  4. python-递归的实现

    一.概念 递归算法是一种直接或者间接地调用自身算法的过程,在计算机编写程序中,递归算法对解决一大类问题是十分有效的. 特点: ①递归就是在过程或者函数里调用自身. ②在使用递归策略时,必须有一个明确的 ...

  5. javascript入门教程 (2)

    这篇我就不铺垫和废话了,我们开始正式进入JS核心语法的学习… 首先我们从基础入手... 一. 基础语法 1.1 区分大小写 JS语法规定变量名是区分大小写的 比如: 变量名 learninpro 和变 ...

  6. EDA风格与Reactor模式

    本文将探讨如下几个问题: Event-Driven架构风格的约束 EDA风格对架构属性的影响 Reactor架构模式 Reactor所解决的问题 redis中的EventDriven 从观察者模式到E ...

  7. Maven--archetypeCatalog笔记

    当我们使用maven原型生成项目骨架时,经常会在[INFO] Generating project in Interactive mode这个地方特别慢,这里并不是什么出错卡住的原因,你打开mvn的d ...

  8. JAVA揭竿而起总要有名号

    古代揭竿而起总要有个响亮的名号,这可不是随便的哦,比如  苍天已死,黄天当立... 玩JAVA里面形形色色的名字,都是有套路的,至于名字怎么起法,那得问问标识符 标识符 用作给变量.类和方法命名.注意 ...

  9. linuxc - entos 7.3 开放端口并对外开放

    1. 查看已打开的端口 # netstat -anp 2. 查看想开的端口是否已开 # firewall-cmd --query-port=666/tcp 若此提示 FirewallD is not ...

  10. bootstrap-01-学习记录

    1.bootstrap所有插件依赖JQ,必须在JQ之后引入. 2.bootstrap分预编译版(css,js,fonts)和源码版(less,js,fonts,dist->预编译版内容,docs ...