文档章节

Android中MVP模式

恰同学少年
 恰同学少年
发布于 2016/02/04 17:12
字数 1528
阅读 345
收藏 12

MVP是MVC的衍生版本,跟MVC类似,但是在Android中更适用,也分三层: 

Model:用于数据的增删改查等,也包括一些数据对象 

View:用于界面的显示与用户操作的接收,在Android里面View通常就是Actvitiy,Fragment。 

Presenter:是View跟Model的“中间人”,接收View的请求后,从Model获取数据交给View。


MVP&MVC的区别

传统MVC有着悠久的历史,但是Android却选用MVP,这绝非偶然。 

在MVC中: 

M:解决用什么去渲染 

V:解决怎么去渲染 

C:解决用户输入事件

在Android中,假如使用MVC,代表V的layout资源并不能完全解决怎么去渲染的问题,还需要Activity的辅助,所以Activity也必然要代表V;但是同时,Activity一方面拥有生命周期回调,另外一方面还为View设置监听,Activity接收来自用户的输入,所以Activity也必然也代表C,Activity就像一个万能对象同时代表着V与C。因此,使用MVC并不能很好地将V与C分离开来。 

与MVC不同的是,MVP中的View是可以接收用户输入,同时也能解决怎么去渲染的问题,所以Activity可以作为MVP里面的View,但是却做不了MVC里面的View。因此,MVP更适用于Android。 

但是View的改变不只在此:View在一方面增加了接收事件的责任,又在另一方面减少了操作Model的责任。View不再直接操作Model,只能通过Presenter去Model操作数据。可以说,MVC中的View与Control交换了部分工作,就成了现在的MVP。

简单例子:

效果图:

项目结构图:

(一)Model

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

package com.zhy.blogcodes.mvp.bean;

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;

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;

public class UserBiz implements IUserBiz {

	@Override
	public void login(final String username, final String password, final OnLoginListener loginL                                    istener) { // 模拟子线程耗时操作
		new Thread() {
			@Override
			public void run() {
				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;

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;

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;

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进行做对应的显示。

© 著作权归作者所有

恰同学少年
粉丝 15
博文 70
码字总数 23613
作品 0
深圳
高级程序员
私信 提问
MVP模式, 开源库mosby的使用及代码分析

Android中的构架模式一直是一个很hot的topic, 近年来Architecture components推出之后, MVVM异军突起, 风头正在逐渐盖过之前的MVP. 其实我觉得MVP还是有好处的, 比如灵活多变(其实只是我用起...

圣骑士wind
2018/09/25
0
0
聊聊Android开发中的MVP模式

一、初识MVP(Model View Presenter) google上关于MVP模式的资料已经特别多了,所以我这里也不啰嗦了。 因为之前做过的几个项目,每个Activity的所有操作代码全部都是堆在里面,虽然查找还算方...

silencezwm
2018/07/03
0
0
MVP架构在xamarin android中的简单使用

好几个月没写文章了,使用xamarin android也快接近两年,还有一个月职业生涯就到两个年了,从刚出来啥也不会了,到现在回头看这个项目,真jb操蛋(真辛苦了实施的人了,无数次吐槽怎么这么丑...

操张林
2018/05/17
0
0
Android之MVVM架构指南(一):导语

之前我在 Android MVP升级路系列的最后一篇文章中提到过有关MVP的没落,虽然有些无奈但没办法这是这样的时代,技术亦如此更何况我们人呢。 很荣幸的是MVP系列的文章得到了好多朋友的回复,其...

吴七禁
2018/10/22
0
0
我的Android重构之旅:架构篇

EF A舞蹈服 去年10月底来到了新公司,刚开始接手 Android 项目时,发现该项目真的是一团遭,项目开发上没有任何架构可言,开发人员连简单的 MVC、MVP 都不了解,Activity 及其臃肿,业务边界...

codeGoogle
2018/05/31
0
0

没有更多内容

加载失败,请刷新页面

加载更多

【AI实战】手把手教你深度学习文字识别(文字检测篇:基于MSER, CTPN, SegLink, EAST等方法)

文字检测是文字识别过程中的一个非常重要的环节,文字检测的主要目标是将图片中的文字区域位置检测出来,以便于进行后面的文字识别,只有找到了文本所在区域,才能对其内容进行识别。 文字检...

雪饼
今天
16
0
思维导图XMind 8 Pro 绿化方法(附序列号)

按部就班: Step 1 -全新下载最新版本的 Xmind 8(注必须是英文官方的版本,中文代{过}{滤}理网站的版本修改过,无法使用pj); Step 2 -安装完毕后,点击文末的下载按钮下载pj补丁文件包,将...

一只小青蛙
今天
10
0
数据结构(ER数据库)设计规范

表命名规范 表命名的规则分为3个层级,层级之间通过_分割,例如b_r_identity、d_l_identity。规约为: [leavel]_[type]_[name] [leavel] 表示数据库表的层级和功能,分为: s:业务无关的系统...

随风溜达的向日葵
今天
10
0
阿里Sentinel控制台源码修改-对接Apollo规则持久化

https://github.com/alibaba/Sentinel/wiki/%E5%9C%A8%E7%94%9F%E4%BA%A7%E7%8E%AF%E5%A2%83%E4%B8%AD%E4%BD%BF%E7%94%A8-Sentinel 动态规则扩展 https://github.com/alibaba/Sentinel/wiki......

jxlgzwh
昨天
14
0
在Linux系统中创建SSH服务器别名

如果你经常通过 SSH 访问许多不同的远程系统,这个技巧将为你节省一些时间。你可以通过 SSH 为频繁访问的系统创建 SSH 别名,这样你就不必记住所有不同的用户名、主机名、SSH 端口号和 IP 地...

老孟的Linux私房菜
昨天
16
1

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部