说到MVP就不得不提到MVC,做过J2EE的猿友们肯定知道MVC是个什么东西。MVC即 Model、View、Controller, 那MVP就Model、View、Presenter。Model用于提供数据模型,View用于显示数据,当然Presenter也就用来处理业务逻辑并将数据显示数据到View上了,它是Model和View的桥梁。

题外话,其实在15年的时候,我们公司的项目就采用mvp,当时没时间写,最近看时间不紧,就分享下

我将整个项目分为三个Module,如下图

其实这种结构我是不太喜欢的,我喜欢在app的根目录下,新建一个liabary文件,这里放第三方的库工程;

一、数据处理模块domain(包含Model),这里的domain命名只是我喜欢这样把数据处理相关的东西都放在这里。

ServiceManager用于向外提供数据的入口(其他类代码在前面博文中已提及)

[java] view
plain
 copy

  1. package com.micky.retrofitrxandroiddragger2.domain.service;
  2. import retrofit.GsonConverterFactory;
  3. import retrofit.Retrofit;
  4. import retrofit.RxJavaCallAdapterFactory;
  5. public class ServiceManager {
  6. private static final String ENDPOINT = "http://ip.taobao.com";
  7. private static class ServiceManagerHolder {
  8. private static final ServiceManager INSTANCE = new ServiceManager();
  9. }
  10. private ServiceManager() {}
  11. public static final ServiceManager getInstance() {
  12. return ServiceManagerHolder.INSTANCE;
  13. }
  14. private ApiService mApiService = null;
  15. public ApiService getApiService() {
  16. if (mApiService == null) {
  17. Retrofit retrofit = new Retrofit.Builder()
  18. .baseUrl(ENDPOINT)
  19. .addConverterFactory(GsonConverterFactory.create())
  20. .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
  21. .build();
  22. mApiService = retrofit.create(ApiService.class);
  23. return mApiService;
  24. }
  25. return mApiService;
  26. }
  27. }

注:我们在这个manager里面主要做一些网络请求,业务处理,数据组装...

二、View和Presenter模块

为了方便管理我将View和Presenter相关类都放在Presentation模块中

View接口

[java] view
plain
 copy

  1. public interface MainView {
  2. void showProgress();
  3. void hideProgress();
  4. void setIpText(String text);
  5. }

Presenter接口

[java] view
plain
 copy

  1. package com.micky.retrofitrxandroiddragger2.presenter;
  2. public interface MainPresenter extends BasePresenter {
  3. void getIpInfo(String ip);
  4. }

Presenter实现类

[java] view
plain
 copy

  1. package com.micky.retrofitrxandroiddragger2.presenter.impl;
  2. import android.text.TextUtils;
  3. import android.util.Log;
  4. import android.widget.Toast;
  5. import com.micky.retrofitrxandroiddragger2.BaseApplication;
  6. import com.micky.retrofitrxandroiddragger2.R;
  7. import com.micky.retrofitrxandroiddragger2.domain.service.ServiceManager;
  8. import com.micky.retrofitrxandroiddragger2.domain.service.response.GetIpInfoResponse;
  9. import com.micky.retrofitrxandroiddragger2.presenter.MainPresenter;
  10. import com.micky.retrofitrxandroiddragger2.presenter.impl.BasePresenterImpl;
  11. import com.micky.retrofitrxandroiddragger2.ui.view.MainView;
  12. import rx.Subscriber;
  13. import rx.android.schedulers.AndroidSchedulers;
  14. import rx.schedulers.Schedulers;
  15. public class MainPresenterImpl extends BasePresenterImpl implements MainPresenter {
  16. private static final String TAG = "TAG";
  17. private MainView mMainView;
  18. public MainPresenterImpl(MainView mainView) {
  19. mMainView = mainView;
  20. }
  21. @Override
  22. public void getIpInfo(String ip) {
  23. if (TextUtils.isEmpty(ip)) {
  24. Toast.makeText(BaseApplication.getContext(), R.string.input_tip_ip, Toast.LENGTH_SHORT).show();
  25. return;
  26. }
  27. mMainView.setIpText("");
  28. mMainView.showProgress();
  29. ServiceManager.getInstance().getApiService().getIpInfo(ip)
  30. .subscribeOn(Schedulers.io())
  31. .observeOn(AndroidSchedulers.mainThread())
  32. .subscribe(new Subscriber<GetIpInfoResponse>() {
  33. @Override
  34. public void onCompleted() {
  35. mMainView.hideProgress();
  36. }
  37. @Override
  38. public void onError(Throwable e) {
  39. Log.e(TAG, e.getMessage(), e);
  40. mMainView.hideProgress();
  41. mMainView.setIpText(BaseApplication.getContext().getString(R.string.network_error));
  42. }
  43. @Override
  44. public void onNext(GetIpInfoResponse getIpInfoResponse) {
  45. mMainView.setIpText(getIpInfoResponse.data.country + " " + getIpInfoResponse.data.area);
  46. }
  47. });
  48. }
  49. }
  50. MainActivity

    [java] view
    plain
     copy

    1. package com.micky.retrofitrxandroiddragger2.ui.activity;
    2. import android.os.Bundle;
    3. import android.support.design.widget.FloatingActionButton;
    4. import android.support.v7.app.AppCompatActivity;
    5. import android.support.v7.widget.Toolbar;
    6. import android.view.View;
    7. import android.view.Menu;
    8. import android.view.MenuItem;
    9. import android.widget.EditText;
    10. import android.widget.ProgressBar;
    11. import android.widget.TextView;
    12. import com.micky.retrofitrxandroiddragger2.R;
    13. import com.micky.retrofitrxandroiddragger2.presenter.MainPresenter;
    14. import com.micky.retrofitrxandroiddragger2.presenter.impl.MainPresenterImpl;
    15. import com.micky.retrofitrxandroiddragger2.ui.view.MainView;
    16. public class MainActivity extends AppCompatActivity implements MainView {
    17. private EditText mEtIp;
    18. private TextView mTvContent;
    19. private ProgressBar mProgressBar;
    20. private MainPresenter mMainPresenter;
    21. @Override
    22. protected void onCreate(Bundle savedInstanceState) {
    23. super.onCreate(savedInstanceState);
    24. setContentView(R.layout.activity_main);
    25. Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
    26. setSupportActionBar(toolbar);
    27. mEtIp = (EditText) findViewById(R.id.et_ip);
    28. mTvContent = (TextView) findViewById(R.id.tv_content);
    29. mProgressBar = (ProgressBar) findViewById(R.id.progress_bar);
    30. mMainPresenter = new MainPresenterImpl(this);
    31. FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
    32. fab.setOnClickListener(new View.OnClickListener() {
    33. @Override
    34. public void onClick(View view) {
    35. mMainPresenter.getIpInfo(mEtIp.getText().toString());
    36. }
    37. });
    38. }
    39. @Override
    40. public boolean onCreateOptionsMenu(Menu menu) {
    41. getMenuInflater().inflate(R.menu.menu_main, menu);
    42. return true;
    43. }
    44. @Override
    45. public boolean onOptionsItemSelected(MenuItem item) {
    46. int id = item.getItemId();
    47. if (id == R.id.action_settings) {
    48. return true;
    49. }
    50. return super.onOptionsItemSelected(item);
    51. }
    52. @Override
    53. public void showProgress() {
    54. mProgressBar.setVisibility(View.VISIBLE);
    55. }
    56. @Override
    57. public void hideProgress() {
    58. mProgressBar.setVisibility(View.GONE);
    59. }
    60. @Override
    61. public void setIpText(String text) {
    62. mTvContent.setText(text);
    63. }
    64. }

    OK,代码基本完了,看了上述代码大家也许都会说以前就在一个类里面就搞定的功能,现在怎么多出这么多接口、实现类啊。别急我刚看到这个mvp的时候也这么想,几度写着写着都把它抛之脑后,这类都多得看不过来了。

    仔细研究下不难发现这M、V、P三者的关系还是挺清晰的。

    好了就到这里,你也赶快试试吧

android 应用模式之mvp的更多相关文章

  1. Android MVC模式和MVP模式的区别

    MVC模式: 1. MVC的所有通信都是单向的. 2. view传送指令到controller(用户也可以直接将指令传到controller). 3. controller完成业务逻辑后要求model ...

  2. 【Android - 框架】之MVP模式的使用

    提起MVP架构模式,大家可能首先想到的是它的"前辈"MVC模式.MVC由Model.View.Controller组成,请求从Controller进入后进行业务判断,然后交给Mod ...

  3. 转:Android开发中的MVP架构(最后链接资源不错)

    Android开发中的MVP架构 最近越来越多的人开始谈论架构.我周围的同事和工程师也是如此.尽管我还不是特别深入理解MVP和DDD,但是我们的新项目还是决定通过MVP来构建. 这篇文章是我通过研究和 ...

  4. Android 架构艺术之MVP

    MVP是Google官方发布的Android开发相关的架构知识.本文要讲解的是一种最基本的MVP的实现方式,它使用手动的依赖注入来提供具有本地和远程数据源的存储库.异步任务处理回调. 基本的MVP的项 ...

  5. 用户界面编程模式 MVC MVP MVVM

    用户界面编程模式 MVC MVP MVVM 程序 = 数据 + 算法 数据:就是待处理的东西 算法:就是代码 涉及到人机交互的程序,不可避免涉及到界面和界面上显示的数据原始方式是界面代码和逻辑代码糅合 ...

  6. android 启动模式介绍

    Android启动模式 (1)Task:与Android系统是个多任务的系统中的任务是不同的.后者更倾向于多进程和多线程来说的,而这里的任务与application(应用程序)和activity(活动 ...

  7. android夜间模式实现

    一.概述 android夜间模式实现分为两大类 重启activity的实现 不重启activity的实现 二.正文 1.重启activity实现夜间模式[在界面文件中的实现部分] 1.1在attrs. ...

  8. Android MVC模式

    Android MVC模式 下面是我对Android MVC模式的理解 Model 模型层 包括实体模型层,存放程序中调用的实体. 业务模型层,存放程序中调用的业务逻辑.   View 显示层  An ...

  9. android recovery模式及ROM制作

    转自android recovery模式及ROM制作 1.总述 为了方便客户日后的固件升级,本周研究了一下android的recovery模式.网上有不少这类的资料,但都比较繁杂,没有一个系统的介绍与 ...

随机推荐

  1. 如何使用《DB 查询分析器》高效地生成旬报货运量数据

    如何使用<DB 查询分析器>高效地生成旬报货运量数据 马根峰                    (广东联合电子服务股份有限公司, 广州 510300) 1      引言   中国本土 ...

  2. 一套强大的vim配置文件+详细注释

    phpchina折腾王独家配置,灰常牛叉的一套vim配置,另附有详细注释,自己折腾vim的时候可以参照其中的大部分设置进行一些个性化定制."是否兼容VI,compatible为兼容,noco ...

  3. iOS 10 推送全解析,注意事项

    本文旨在对 iOS 推送进行一个完整的剖析,如果你之前对推送一无所知,那么在你认真地阅读了全文后必将变成一个推送老手,你将会对其中的各种细节和原理有充分的理解.以下是 pikacode 使用 iOS ...

  4. 【Netty源码学习】BootStrap

    BootStrap是客户端的启动类,其主要功能就是设置必要的参数然后启动客户端. 实现如下: Bootstrap b = new Bootstrap(); b.group(group) .channe ...

  5. Android开发学习之路--Activity之Intent

    窗外再次飘起了小雪,还有1周就过年了,2016年即将到来,来年不知道自己将身处何处,船到桥头自然直吧.还是继续学习吧,上次学习了Activity,那么如果是两个Activity之间,怎么从一个Acti ...

  6. Android开发学习之路--Android系统架构初探

    环境搭建好了,最简单的app也运行过了,那么app到底是怎么运行在手机上的,手机又到底怎么能运行这些应用,一堆的电子元器件最后可以运行这么美妙的界面,在此还是需要好好研究研究.这里从芯片及硬件模块-& ...

  7. python的operator.itemgetter('click')用于定义获取'click'项的函数

    python的排序参见文章http://blog.csdn.net/longshenlmj/article/details/12747195 这里介绍 import operator模块 operat ...

  8. 1022. Digital Library (30) -map -字符串处理

    题目如下: A Digital Library contains millions of books, stored according to their titles, authors, key w ...

  9. TCP模型及其重点协议总结

    概述 TCP/IP协议族,作为最早的协议模型(后来OSI七层也是在该基础上细分而来),每层都有一些重点的协议,面试时也会被询问,快要找工作,得做一些总结了 [1]TCP4层协议模型概述 [2]各层重点 ...

  10. C++ Primer 有感(重载操作符)

    1.用于内置类型的操作符,其含义不能改变.也不能为任何内置类型定义额外的新的操作符.(重载操作符必须具有至少一个类类型或枚举类型的操作数.这条规则强制重载操作符不能重新定义用于内置类型对象的操作符的含 ...