MVP for Android

作者: Yink_Liu | 来源:发表于2017-03-04 21:42 被阅读379次

    注:便于更好的理解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

    相关文章

      网友评论

        本文标题:MVP for Android

        本文链接:https://www.haomeiwen.com/subject/ycqcpttx.html