MVP是MVC的衍生版本,跟MVC相似,可是在Android中更适用,也分三层: java
Model:用于数据的增删改查等,也包括一些数据对象 android
View:用于界面的显示与用户操做的接收,在Android里面View一般就是Actvitiy,Fragment。 服务器
Presenter:是View跟Model的“中间人”,接收View的请求后,从Model获取数据交给View。app
MVP&MVC的区别ide
传统MVC有着悠久的历史,可是Android却选用MVP,这绝非偶然。 post
在MVC中: this
M:解决用什么去渲染 spa
V:解决怎么去渲染 线程
C:解决用户输入事件code
在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进行作对应的显示。