注:便于更好的理解MVP并区分和MVC的差别,推荐阅读我另一篇文章MVC for Android
概念
MVP,即Model(模型)、View(视图)、Presenter(主持人)。MVP是从经典的MVC演变而来。把数据处理,界面显示,逻辑处理分离开来。界面和数据的所有通信都是通过P层来实现。是一个将后台任务和activities/views/fragment分离的方法,让它们独立于绝大多数跟生命周期相关的事件。
MVP for Android
理解MVP:
View:与用户交互,响应用户的操作,显示数据,在Android中,通常是Activity、Fragment,View等;
Presenter:控制层,也负责处理后台任务;
Model:数据层,比如数据库接口,网络数据等;
例子:User现在要搜索联系人名字为Jack的号码是多少(和MVC for Android例子一样方便读者理解对比)
MVP特点:
1、模型与视图完全分离,我们可以修改视图而不影响模型
2、可以更高效地使用模型,因为所有的交互都发生在一个地方——Presenter内部
3、我们可以将一个Presenter用于多个视图,而不需要改变Presenter的逻辑。这个特性非常的有用,因为视图的变化总是比模型的变化频繁。
4、如果我们把逻辑放在Presenter中,那么我们就可以脱离用户接口来测试这些逻辑(单元测试)
实例代码
上述例子截图
View
本例子Activity充当View,布局如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:orientation="vertical"
tools:context="example.yink.mvp.MainActivity">
<EditText
android:id="@+id/et_name"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/et_name"/>
<Button
android:id="@+id/bt_search"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/btn_search"/>
<TextView
android:id="@+id/tv_numbere"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/tv_num"/>
</LinearLayout>
再看Activity代码,主要是布局view
public class MainActivity extends AppCompatActivity implements View.OnClickListener, SearchViewInterface {
private EditText mEditTextName;
private Button mButtonSearch;
private TextView mTExtViewNum;
private SearchPresenter mSearchPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
mSearchPresenter = new SearchPresenter();
mSearchPresenter.attachView(this);
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_CONTACTS}, 1);
}
private void initView() {
mEditTextName = (EditText) findViewById(R.id.et_name);
mButtonSearch = (Button) findViewById(R.id.bt_search);
mTExtViewNum = (TextView) findViewById(R.id.tv_numbere);
mButtonSearch.setOnClickListener(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
mSearchPresenter.detachView();
}
@Override
public void onSuccess(String num) {
mTExtViewNum.setText(num);
}
@Override
public void onError() {
Toast.makeText(this,"no number",Toast.LENGTH_SHORT).show();
}
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.bt_search:
mSearchPresenter.searchNum(mEditTextName.getText().toString());
break;
}
}
}
Activity和Presenter的衔接主要是通过SearchNumListener,当Presenter控制调取数据或者其他长时间操作后,通过此借口回调反馈给View。
public interface SearchViewInterface {
void onSuccess(String num);
void onError();
}
Presenter
控制层,它主要功能是和View以及Model进行功能对接。
public class SearchPresenter implements SearchModelInterface{
private MainActivity mView;
private SearchNumModelImpl mSearchNumModelImpl;
public void attachView(MainActivity view) {
this.mView = view;
}
public void detachView() {
this.mView = null;
}
public void searchNum(String name) {
if (mSearchNumModelImpl == null) {
mSearchNumModelImpl = new SearchNumModelImpl(mView);
}
}
@Override
public void getNum(String num) {
if (Util.isEmptyString(num)) {
mView.onError();
} else {
mView.onSuccess(num);
}
}
}
这里也是利用接口SearchModelInterface链接Model和Presenter,Model可以是数据查询,耗时网络操作等。等有结果后,SearchModelInterface反馈给Presenter,控制层接收到数据后,反馈给View
public interface SearchModelInterface {
void getNum(String num);
}
Model
数据独立查询
public class SearchNumModelImpl{
private Context mContext;
public SearchNumModelImpl(Context context) {
mContext = context;
}
public void getNum(String name,SearchModelInterface mSearchModelInterface) {
if (!Util.isEmptyString(name)) {
mSearchModelInterface.getNum(number(name));
}
}
private String number(String name) {
Cursor cursor = mContext.getContentResolver().query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null);
while (cursor.moveToNext()) {
String contactId = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID));
String contactName = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
if (name.equals(contactName)) {
Cursor phone = mContext.getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = " + contactId, null, null);
if (phone.moveToNext()) {
String phoneNumber = phone.getString(phone.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
return phoneNumber;
}
}
}
return null;
}
}
整个过程如下:
View:Activity充当View,View和用户交互,通知Presenter记性查询操作
Presenter:Presenter接到View的消息,控制查询开始。等待接收Model返回数据。返回后再通知View显示。Control
Model:详细数据查询操作封装,查询到数据后,反馈给Presenter
总结
此例子和笔者另一文MVC for Android,例子相同。结构清晰,两篇文章都是一个对框架的一个简单理解。实际运用在项目中还需要更多的细节注意。内存泄露,nullpointerexception等等。目的主要理解MVP的作用和意义。
demo源码github,分支demo-mvp android-architecture
网友评论