MVP实战 -- 第二篇

这篇也是我在网上看见的一篇文章,觉得写得不错。认真学习了,也敲了几遍。有了自己的理解,现在按照自己的思路来阐述一下。原文地址:http://mp.weixin.qq.com/s/opTZJT80s2VEZUTJaAyo9g

整体逻辑

模拟请求数据并用ListView显示,请求的时候会有ProgressBar提示,2秒后ProgressBar小时。点击ListView的item吐司提示点击了那个item。


图一

老规矩先分包

图二

先从简单的View入手吧,分析了前面的逻辑,UI方面提取出四个 抽象方法

图二

有了这个接口,只要让我们的界面去实现这个接口即可。

图三

这个后面会详细分析。
然后就是Model层,先定义一个请求的接口:

图四

注意里面的参数是一个接口回调,用来回调我们模拟网络请求的数据。

图五

网络请求无非两种结果,请求成功和请求失败,请求成功后当然会有数据传递过来,然后我们根据相应的数据来做相应的操作。
定义网络请求的接口当然要创建一个类去实现这个接口,要不然接口就毫无意义了。

图六

在这里面我们创建一个线程,并然这个线程休眠2秒,模拟请求的耗时,将模拟的数据通过OnRequestListener回调回去,我们只需要在其onSuccess()中做相应的操作即可。
最后就是Presenter了,所有的逻辑处理都在这里面。太长了,不好截图我就直接上代码了。

public class RequestBizPresenter{
    private IView mIView;
    private RequestBiz mRequestBiz;
    private Handler mHandler;

    public RequestBizPresenter(IView IView) {
        this.mIView = IView;
        mIView.showProgressBar();
        mRequestBiz = new RequestBizImpl();
        mHandler = new Handler(Looper.getMainLooper());
    }

    public void requestForData() {  //接口回调
        mRequestBiz.requestForData(new OnRequestListener() {
            @Override
            public void onSuccess(final List<String> data) {
                //是在子线程中或得到data的,需要在主线程中去更新UI
                mHandler.post(new Runnable() {
                    @Override
                    public void run() {
                        //请求成功的回调
                        mIView.hideProgressBar();
                        mIView.showMsg("请求成功");
                        mIView.setListItem(data);
                    }
                });
            }

            @Override
            public void onFailed() {
                //请求失败的回调
                mIView.showMsg("请求失败");
            }
        });
    }

    public void showItemClick(int position) {
        mIView.showMsg("点击了第" + position + "个item");
    }
}

作为View和Model的中间人,当然必须要将他们两个都叫到一块来(即拥有两者,作为成员变量)。

public RequestBizPresenter(IView IView) {
        this.mIView = IView;
        mIView.showProgressBar();
        mRequestBiz = new RequestBizImpl();
        mHandler = new Handler(Looper.getMainLooper());
    }

传递进来的IView其实就是这个接口的实现类,在这里是MAinActivity,记住接口是不能实例化的。注意我们的回调方法是在子线程,我们要通过Handler,post到主线程去更新UI。这里面的逻辑也就是两个,一个是请求数据,二是判断哪个被点击,调用这个showItemClick()方法即可。
最后就是实现了IView接口的MAinActivity了

public class MainActivity extends AppCompatActivity implements IView, AdapterView.OnItemClickListener {
    private RequestBizPresenter mPresenter;
    private ListView mListView;
    private ProgressBar mProgressBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
//        mPresenter = new RequestBizPresenter(this);  TODO 注意实例化的位置
        mListView = (ListView) findViewById(R.id.listView);
        mProgressBar = (ProgressBar) findViewById(R.id.progressbar);
        mListView.setOnItemClickListener(this);
//        mPresenter.requestForData();  TODO 注意实例化的位置
        mPresenter = new RequestBizPresenter(this);
        mPresenter.requestForData();
    }


    @Override
    public void showProgressBar() {
        mProgressBar.setVisibility(View.VISIBLE);
    }

    @Override
    public void hideProgressBar() {
        mProgressBar.setVisibility(View.GONE);
    }

    @Override
    public void showMsg(String msg) {
        Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
    }

    @Override
    public void setListItem(List<String> data) {
        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,data);
        mListView.setAdapter(adapter);
    }


    @Override
    public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
        mPresenter.showItemClick(i);
    }
}

在这里我们只是进行UI的处理,逻辑全部交给Presenter去实现。
我在这里粗心还犯了一个错误,可以看代码的注释部分,一定要注意逻辑的先后顺序,要不然会出现空指针。
是不是看到MainActivity很清爽,没有其他乱七八糟的代码。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容