本文是参考google官方发布的MVP架构demo以及前人对MVP实现方式的一些总结做的一个简单demo,在这里记录一下一点心得,希望能给想用MVP的人一点帮助。
## 总体框架 ### 工程目录结构 首先先看下整个工程的目录结构: 目录的代码组织方式是按照功能来组织的,功能内部分为xactivity、xcontract、xfragment、xpresenter四个类文件(x代表业务名称)。base文件夹存放一些公用的基类文件,data文件夹存放业务逻辑相关的代码,utils文件夹则放一些公用的工具类。本demo实现的功能为:通过点击界面上的按钮,获取手机相关信息,获取过程中加入延时及等待提示(模拟网络),最终将信息显示于界面上(简单演示,只是显示了系统时间)。
基类
先看下BasePresenter与BaseView这两个接口类,它们分别是所有Presenter与View的基类。
java
public interface BasePresenter {
void start();
}
BasePresenter中含有方法start(),该方法的作用是presenter开始获取数据并调用view中方法改变界面显示,其调用时机是在Fragment类的onResume方法中。
public interface BaseView<T> {
void setPresenter(T presenter);
}
BaseView中含方法setPresenter,该方法作用是将presenter实例传入view中,其调用时机是在activity的presenter实现类的构造函数中。
契约类
public interface GetPhoneInfoContract {
interface View extends BaseView<Presenter> {
void setTime(String time);
void showLoading();
void hideLoading();
}
interface Presenter extends BasePresenter {
void getTime();
}
}
与比较常见的mvp实现不同,官方的实现中加入了契约类来统一管理view与presenter的所有的接口,这种方式使得view与presenter中有哪些功能,一目了然,维护起来也方便,该实例中presenter的接口实现获取系统时间,view的接口实现时间的显示以及提示对话框的显示及隐藏。
MVP的组织方式
### activity的作用 activity作为全局的控制者,负责创建view以及presenter实例,并将二者联系起来,具体的view交由fragment来实现,两者各司其职。
@EActivity(R.layout.get_phone_info_act)
public class GetPhoneInfoActivity extends ActionBarActivity {
private FragmentManager fm;
private GetPhoneInfoFragment mGetPhoneInfoFragment = new GetPhoneInfoFragment_();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setDefaultFragment();
new GetPhoneInfoPresenter(mGetPhoneInfoFragment);
}
private void setDefaultFragment() {
fm = getFragmentManager();
FragmentTransaction transaction = fm.beginTransaction();
transaction.add(R.id.fragcontent, mGetPhoneInfoFragment);
transaction.commit();
}
}
本例中,activity中通过setDefaultFragment()设置了fragment,之后实例化GetPhoneInfoPresenter,并将frament传递进去,实现在presenter中通过fragment的接口对view进行操作展示。
presenter的实现
public class GetPhoneInfoPresenter implements GetPhoneInfoContract.Presenter{
private final GetPhoneInfoContract.View mGetPhoneInfoView;
private PhoneInfoBiz phoneInfoBiz;
public GetPhoneInfoPresenter(GetPhoneInfoContract.View getPhoneInfoView) {
mGetPhoneInfoView = getPhoneInfoView;
mGetPhoneInfoView.setPresenter(this);
phoneInfoBiz = new PhoneInfoBizIml();
}
@Override
public void start() {
getTime();
}
@Override
public void getTime() {
mGetPhoneInfoView.showLoading();
phoneInfoBiz.getPhoneInfo(new PhoneInfoBiz.GetPhoneInfoCallback() {
@Override
public void onGetPhoneInfo(PhoneInfo phoneInfo) {
mGetPhoneInfoView.setTime(phoneInfo.getTime());
mGetPhoneInfoView.hideLoading();
}
});
}
}
presenter构造函数中调用了view的setPresenter方法将自身实例传入,start方法中处理了数据加载与展示。如果需要界面做对应的变化,直接调用view层的方法即可,这样view层与presenter层就能够很好的被划分。
view的实现
@EFragment(R.layout.get_phone_info_frag)
public class GetPhoneInfoFragment extends Fragment implements GetPhoneInfoContract.View {
private GetPhoneInfoContract.Presenter mPresenter;
ProgressDialog dialog;
@ViewById
TextView tv_time;
@ViewById
Button btn_get_time;
@Click
void btn_get_time() {
mPresenter.getTime();
}
@AfterViews
void initView() {
dialog = new ProgressDialog(getActivity());
}
@Override
public void onResume() {
super.onResume();
mPresenter.start();
}
@Override
public void setPresenter(GetPhoneInfoContract.Presenter presenter) {
if (presenter != null)
mPresenter = presenter;
}
@Override
@UiThread
public void setTime(String time) {
tv_time.setText(time);
}
@Override
public void showLoading() {
dialog.setTitle("请稍候");
dialog.setMessage("loading!");
dialog.show();
}
@Override
public void hideLoading() {
dialog.dismiss();
}
}
setPresenter方法继承于父类,通过该方法,view获得了presenter得实例,从而可以调用presenter代码来处理业务逻辑。在onResume中还调用了presenter得start方法,处理数据的加载与展示。
简单归纳
- activity创建fragment及实例化presenter,在实例化的同时,将fragment作为参数传递进去,这样一来在presenter中就可调用view的接口对界面进行更新、展示
- presenter实例化时,还调用了view的setPresenter方法,将自身传递进去,这样一来fragment获得了presenter的实例,便可在view中调用presenter进行业务逻辑的操作
- view及presenter拥有彼此的实例,实现了在view中调用presenter处理业务,处理完后再presenter中更新view。
model层
简单介绍下model层,PhoneInfo对象存储手机相关信息,PhoneInfoBiz为借口类实现该业务所需要的接口及回调接口,PhoneInfoBizIml为接口的实现类。直接贴代码:
public interface PhoneInfoBiz {
interface GetPhoneInfoCallback {
void onGetPhoneInfo(PhoneInfo phoneInfo);
}
void getPhoneInfo(GetPhoneInfoCallback getPhoneInfoCallback);
}
public class PhoneInfoBizIml implements PhoneInfoBiz{
@Override
public void getPhoneInfo(final GetPhoneInfoCallback getPhoneInfoCallback) {
new Thread(new Runnable() {
@Override
public void run() {
try {
PhoneInfo phoneInfo = new PhoneInfo();
phoneInfo.setTime(System.currentTimeMillis() + "");
phoneInfo.setMobileType(Build.MODEL);
phoneInfo.setMobileVer(Build.VERSION.RELEASE);
Thread.sleep(1000);
getPhoneInfoCallback.onGetPhoneInfo(phoneInfo);
}catch(Exception e){
e.printStackTrace();
}
}
}).start();
}
}
总结
至此,一个简单的mvp框架到此结束,对于mvp的使用目前也还在探索中,上例是结合官方发布的demo做的一个简化工程,有不足之处欢迎一起探讨交流!
最后附上本文demo及官方demo的地址: