软件设计模式:基于MVP的Android项目架构

一、概述
每一个 app 的运营都需要经过不断的迭代与更新!在产品不断的升级过程中,项目的代码量会变得越来越大。当采用 Android 原生的架构( Android 项目的架构:当建立起一个新项目时,默认的就像是个 MVC 的架构)去不断完善、升级项目时。到最后项目就变得越来越臃肿,这时,随着项目的越来越大,也许不得已需要进行项目的重构,然而这是个工作量很大的任务。所以,做好项目的架构,写好模型往往是很重要的。

google 官方在 GitHub 上也有着对应的官方规范已完善的开源项目架构(可以直接到这里下载 demo 学习):

  1. todo-mvp/- Basic Model-View-Presenter architecture(基本的 mvp 架构)
  2. todo-mvp-loaders/- Based on todo-mvp, fetches data using Loaders.(基于 todo-mvp ,数据获取采用 Loaders)
  3. todo-databinding/- Based on todo-mvp, uses the Data Binding Library.(基于 todo-mvp ,使用 databinding 开源库)
  4. todo-mvp-clean/- Based on todo-mvp, uses concepts from Clean Architecture.(基于 todo-mvp,使用 Clean 架构)
  5. todo-mvp-dagger/- Based on todo-mvp, uses Dagger2 for Dependency Injection(基于 todo-mvp,使用 dagger2 进行依赖注入)
  6. todo-mvp-contentproviders/- Based on todo-mvp-loaders, fetches data using Loaders and uses Content Providers(基于 todo-mvp-loaders ,数据获取采用 Loaders 和 ContentProviders)
  7. todo-mvp-rxjava/- Based on todo-mvp, uses RxJava for concurrency and data layer abstraction.(基于 todo-mvp ,使用了 Rxjava )

二、Android Architecture Blueprints:todo-mvp 学习
(1)首先,我们来看一下 todo-mvp 的包结构:

包结构
按功能来分包
model层

我们可以看到,在 todo-mvp demo 中,一个包就对应着一个功能模块。在 tasks 包中:

  1. TasksContract(定义 Presenter、View 的接口)
  2. TasksFragment(实现了 TasksContract.View 接口定义的功能)
  3. TasksPresenter(实现了 TasksContract.Presenter 接口定义的功能) 4. TasksActivity(TasksFragment 的载体)

在 data 包中,则是定义了对应的功能需要的数据实体、model 接口:

  1. 任务实体 Task
  2. model 接口 TasksDataSource
  3. 对应功能的 TasksDataSource实现类

(2)源码分析

  1. 首先,我们先看一下Presenter、View的两个基类
public interface BasePresenter { void start();}

start()方法在View界面开始显示的时候调用,一般是在Activity、Fragment的onStart()方法中

public interface BaseView<T> { void setPresenter(T presenter);}

setPresenter(T presenter)方法在Presenter实例化之后调用,一般是在Presenter构造方法最后调用,传入自己。

  1. 我们从 TasksActivity 入手,在 onCreate() 方法中,可以看到下面一段代码
@Override 
protected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); 
...略 
        //添加TasksFragment 
        TasksFragment tasksFragment = (TasksFragment)getSupportFragmentManager().findFragmentById(R.id.contentFrame); 
        if (tasksFragment == null) {
                tasksFragment = TasksFragment.newInstance();
                ActivityUtils.addFragmentToActivity( getSupportFragmentManager(), tasksFragment, R.id.contentFrame); 
        } 
        //新建TasksPresenter实例 
        mTasksPresenter = new TasksPresenter(Injection.provideTasksRepository(getApplicationContext()), tasksFragment);
 }

TasksActivity 中,在添加 TasksFragment 后 new 了一个 TasksPresenter,接下来看TasksPresenter的构造方法。

public TasksPresenter(@NonNull TasksRepository tasksRepository, @NonNull TasksContract.View tasksView) {
         mTasksRepository = checkNotNull(tasksRepository, "tasksRepository cannot be null");
         mTasksView = checkNotNull(tasksView, "tasksView cannot be null!");
         mTasksView.setPresenter(this); 
}

TasksPresenter的构造方法中,TasksPresenter获得了对mTasksView引用,并且还调用了mTasksView的setPresenter()方法。此时,TasksPresenter就传递到了TasksFragment 中,只要在TasksFragment 中对mPresenter 进行赋值,就完成了TasksFragment 与TasksPresenter实例引用的相互持有。

@Override public void setPresenter(@NonNull TasksContract.Presenter presenter) { 
        mPresenter = presenter;
 }

到这里你也许会问,你只说了 V 与 P 的联系,那 M 去哪里了呢?嘿嘿,你在返回去看TasksPresenter的构造方法,这时你是否会发现这一句代码。

 mTasksRepository = checkNotNull(tasksRepository, "tasksRepository cannot be null");

没错,就是在实例化TasksPresenter的同时,将数据处理model TasksRepository 赋予了TasksPresenter。至此,既然知道了各自的引用关系,那View、Presenter、Model的关系也就清楚了。

注意:Presenter的实例化在Activity中,但View对Presenter的引用却不一定在Activity中。因为View的实现类不一定是Activity,也可能是Fragment。至此,我们可以得出一个结论:Presenter的实例化在Activity中,View对Presenter的引用在View的实现类中。

3.既然清楚了View、Presenter、Model之间的关系,那么我们来看看它们之间到底是怎样协作的。由于代码比较多,我们就选择刷新数据功能来讲。

  • View
@Nullable 
@Override 
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        ...略  
       swipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() { 
       @Override
        public void onRefresh() {
                mPresenter.loadTasks(false); 
               }
        });  
       ...略 
       return root; 
}

在刷新时,调用了 mPresenter.loadTasks(false);可以看到,View将刷新功能交给了Presenter来做。再来看看mPresenter.loadTasks(false)方法 - Presenter

@Override 
public void loadTasks(boolean forceUpdate) {
         loadTasks(forceUpdate || mFirstLoad, true);
         mFirstLoad = false; 
} 
 //私有方法,操作数据 
private void loadTasks(boolean forceUpdate, final boolean showLoadingUI) {
         if (showLoadingUI) { 
                //对mTasksView的选中框进行显示  
                mTasksView.setLoadingIndicator(true);
         } 
        if (forceUpdate) {
                //mTasksRepository中的数据操作、刷新数据
                mTasksRepository.refreshTasks(); 
        } 
        EspressoIdlingResource.increment(); 
        // App is busy until further notice 
        //mTasksRepository中的回调方法处理返回的数据
        mTasksRepository.getTasks(new  TasksDataSource.LoadTasksCallback() { 
        @Override 
        public void onTasksLoaded(List<Task> tasks) {
         List<Task> tasksToShow = new ArrayList<Task>();
         // This callback may be called twice, once for the cache and once for loading 
        // the data from the server API, so we check before decrementing, otherwise 
        // it throws "Counter has been corrupted!" exception. 
        if (!EspressoIdlingResource.getIdlingResource().isIdleNow()) { 
                EspressoIdlingResource.decrement();
                 // Set app as idle. 
        } 
        // We filter the tasks based on the requestType
         for (Task task : tasks) { 
         switch (mCurrentFiltering) {
                case ALL_TASKS:
                        tasksToShow.add(task); 
                break; 
                case ACTIVE_TASKS:
                 if (task.isActive()) { 
                         tasksToShow.add(task); 
                }
                break; 
                case COMPLETED_TASKS:
                if (task.isCompleted()) {
                         tasksToShow.add(task);
                } 
                break;
                default: 
                        tasksToShow.add(task); 
                break;
              }
         } 
        // The view may not be able to handle UI updates anymore 
        if (!mTasksView.isActive()) { 
                return;
         }
        if (showLoadingUI) { 
                mTasksView.setLoadingIndicator(false);
        }
           processTasks(tasksToShow); 
        }

        @Override 
        public void onDataNotAvailable() {
                // The view may not be able to handle UI updates anymore
                 if (!mTasksView.isActive()) { 
                        return;
                } 
                mTasksView.showLoadingTasksError();
                } 
        });
 }

mPresenter.loadTasks()方法通过调用一个同名私有的重载的方法,调用 mTasksView.setLoadingIndicator(true)显示是选中框、 mTasksRepository.refreshTasks()刷新数据、 mTasksRepository.getTasks()的回调方法处理返回的数据。

总结:MVP在我看来就是一种“代理模式”。View、Model只需要做好自己的任务,然后,Model处理后的数据交于Presenter,Presenter通过View的引用将处理后的数据进行显示。最后,还有个契约类,也就是Contract,主要用于管理Presenter与View的接口,方便扩展。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容