说到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. Android Studio安装Genymotion插件

    Android Studio安装Genymotion插件 Eclipse就不介绍了,谷歌都已经放弃Eclipse了,你还在坚持什么. 安装Genymotion 官网:https://www.genym ...

  2. Android高级控件(六)——自定义ListView高仿一个QQ可拖拽列表的实现

    Android高级控件(六)--自定义ListView高仿一个QQ可拖拽列表的实现 我们做一些好友列表或者商品列表的时候,居多的需求可能就是需要列表拖拽了,而我们选择了ListView,也是因为使用L ...

  3. 带你深入理解STL之List容器

    上一篇博客中介绍的vector和数组类似,它拥有一段连续的内存空间,并且起始地址不变,很好的支持了随机存取,但由于是连续空间,所以在中间进行插入.删除等操作时都造成了内存块的拷贝和移动,另外在内存空间 ...

  4. RecyclerView嵌套RecyclerView

    ListView嵌套GridView http://blog.csdn.net/baiyuliang2013/article/details/42646289 RecyclerView下拉刷新上拉加载 ...

  5. 07 总结ProgressDialog 异步任务

    1,ProgressDialog     >        //使用对象  设置标题             progressDialog.setTitle("标题");   ...

  6. UNIX网络编程——客户/服务器程序设计示范(三)

    TCP预先派生子进程服务器程序,accept无上锁保护 我们的第一个"增强"型服务器程序使用称为预先派生子进程的技术.使用该技术的服务器不像传统意义的并发服务器那样为每个客户现场派 ...

  7. Android开发学习之路--UI之初体验

    之前都是学习Activity,对于布局都没有做过学习,这里就简单学习下吧.下面看下Android Studio下有哪些控件: 这里分为Widgets,Text Fields,Containers,Da ...

  8. Unity2D Sprite灰白图(Unity3D开发之十一)

    猴子原创,欢迎转载.转载请注明: 转载自Cocos2D开发网–Cocos2Dev.com,谢谢! 原文地址: http://www.cocos2dev.com/?p=596 昨晚看到群里问到2DSpr ...

  9. 一致性Hash算法介绍(分布式环境算法)

    32的整数环(这个环被称作一致性Hash环),根据节点名称的Hash值(其分布范围同样为0~232)将节点放置在这个Hash 环上.然后根据KEY值计算得到其Hash值(其分布范围也同样为0~232  ...

  10. ROS_Kinetic_07 ROS中机器人三维物理引擎高保真仿真利器gazebo 7.0

    ROS_Kinetic_07 ROS中机器人三维物理引擎高保真仿真利器gazebo 7.0 ROS kinetic中的gazebo版本是7.0,有很多新的特性. 首先,启动gazebo: ~$ gaz ...