给大家推荐个开源项目 : T-MVVM

MVP缺点

  1. Presenter中除了逻辑以外,还有大量的View->Model,Model->View的逻辑操作,造成Presenter臃肿,维护困难。

  2. 对UI的渲染放在了Presenter中,所以UI和Presenter的交互会过于频繁。

  3. Presenter过多地渲染了UI,往往会使得它与特定的UI的交互频繁,一旦UI变动,Presenter也需要变。

  4. 接口暴增,可以说代码量成倍增长,交互都需要通过接口传递信息,让人无法忍受。

基本上用过MVP的码友们都能发现了以上诸多弊端,于是小码就尝试从传统的MVP过度到T-MVVM,深度结构,告别繁琐的接传递信息。

优秀的架构能让软件高内聚、低耦合、可维护、可扩展,其实,对于MVP或者MVVM没有绝对好坏,MVP->MVVM只不过让模型和视图分离得更加的彻底,优化成了前者的缺点,如果项目业务不是很多或者业务相对简单,其实完全没有必要使用MVP/MVVM,反而让项目变得更为复杂。

什么是T-MVVM

基于ViewModel、LiveData、Retrofit、OkHttp和Rxjava实现的T-MVVM体系结构的架构,泛型限定,深度解耦。

ViewModel优点: * 同步关联生命周期 * 数据共享 * 复用性强LiveData优点: * 确保UI界面的数据状态 * 没有内存泄漏,不会因为Activity的不可见导致Crash * 不用再人为的处理生命周期 * 共享资源

此架构未使用DataBinding原因:

* 数据绑定增加Bug调试难度* 复杂的页面,model也会很大,虽然使用方便了也很容易保证了数据的一致性,当长期持有,不利于释放内存* 数据双向绑定不利于View重用

T-MVVM的实现细节

MVVM的调用和MVP类似,在MVP中全部由Presenter负责ViewModel之间的数据同步,而MVVM中ViewModel充当了Presenter的角色,ViewModel是View与Model的连接器,持有可被观察的数据持有者和网络请求操作,数据变更实时渲染UI。

如何使用T-MVVM架构

1.先定义BaseViewModel基类

 /**  * @author:tqzhang on 18/7/26 16:15  */  public class BaseViewModel<T extends BaseRepository> extends AndroidViewModel {    public T mRepository;    public BaseViewModel(@NonNull Application application) {        super(application);        mRepository = TUtil.getNewInstance(this, 0);    }    @Override    protected void onCleared() {        super.onCleared();        if (mRepository != null) {           mRepository.unSubscribe();        }   } }

BaseViewModel通过泛型类型参数BaseRepository子类初始化Repository数据仓库,同时在activity/fragment走onDestroy()生命周期方法时 BaseViewModel回调onCleared,即页面销毁是用来取消网络请求或资源释放等操作。

正常开发一般不建议直接通过ViewModel获取网络数据,这里我们将工作交给一个新的模块Repository。Repository只负责数据处理,提供干净的api,同时方便切换数据来源。

2.再定义BaseRepository

public abstract class BaseRepository {     protected ApiService apiService;     public BaseRepository() {             if (null == apiService) {                 apiService = HttpHelper.getInstance().create(ApiService.class);             }     }     private CompositeSubscription mCompositeSubscription;     protected void addSubscribe(Subscription subscription) {            if (mCompositeSubscription == null) {                mCompositeSubscription = new CompositeSubscription();            }           mCompositeSubscription.add(subscription);      }     public void unSubscribe() {             if (mCompositeSubscription != null && mCompositeSubscription.hasSubscriptions()) {            mCompositeSubscription.clear();    }}

BaseRepository中内容相对简单,主要是获取ApiService和网络请求订阅容器,方便管理网络请求,即页面销毁是取消网络请求操作。

3.然后自定义AbsLifecycleFragment基类继承BaseFragment,BaseFragment可自行编写。如不需要使用T-MVVM,可自行继承BaseFragment,互不影响。

 public abstract class AbsLifecycleFragment<T extends BaseViewModel> extends BaseFragment  {       protected T mViewModel;       /**        * init view        * @param state        */       @Override       public void initView(Bundle state) {            mViewModel = VMProviders(this, TUtil.getInstance(this, 0));          if (null != mViewModel) {            dataObserver();          }       }      /**       * create ViewModelProviders       *       * @return ViewModel       */      protected <T extends ViewModel> T VMProviders(BaseFragment fragment, @NonNull Class<T> modelClass) {           return ViewModelProviders.of(fragment).get(modelClass);      }      protected void dataObserver() {      }     /**      * 获取网络数据      */      protected void getRemoteData() {      }     }

在initView方法中通过BaseViewModel子类泛型类型参数获取Class<t style="margin: 0px; padding: 0px; max-width: 100%; box-sizing: border-box !important; word-wrap: break-word !important; font-size: inherit; color: inherit; line-height: inherit;">,在通过ViewModelProviders.of(fragment).get(modelClass))实例化ViewModel,到此我们的基类基本编写完毕。</t>

4.下面我们以一个简单业务实战下,获取文章列表。

4-1:ArticleFragment

/*** @author:tqzhang on 18/7/2 14:40*/public class ArticleFragment extends AbsLifecycleFragment<ArticleViewModel> {   protected TRecyclerView mRecyclerView; protected StaggeredGridLayoutManager layoutManager; protected MultiTypeAdapter adapter; public static ArticleFragment newInstance() {    return new ArticleFragment(); } @Override public int getLayoutResId() {    return R.layout.fragment_list; } @Override public void initView(Bundle state) {    super.initView(state);    mRecyclerView=findViewById(R.id.recycler_view);    initAdapter();    initRecyclerView();    //获取网络数据    getRemoteData(); } public void initRecyclerView(){    layoutManager=new new StaggeredGridLayoutManager(1,  StaggeredGridLayoutManager.VERTICAL);    mRecyclerView.setAdapter(adapter);    mRecyclerView.setLayoutManager(layoutManager);} //初始化adapter public void initAdapter(){   adapter= new MultiTypeAdapter.Builder<>()           .bindArray(ArticleInfoVo.class, new ArticleRem1ItemHolder(context)            , new ArticleRem2ItemHolder(context)            , new ArticleRem3ItemHolder(context))           .bind(HeaderVo.class, new HeaderViewHolder(context, rogressStyle.Pacman))           .bind(FootVo.class, new FootViewHolder(context, ProgressStyle.Pacman))           .build(); //数据观察 @Override protected void dataObserver() {    mViewModel.getArticleList().observe(this, articleVo -> {        if (null != articleVo) {           mRecyclerView.refreshComplete(articleVo.data.list, false);        }    }); } //获取网络数据 @Override protected void getRemoteData() {     mViewModel.getArticleList(typeId, lastId); }}

我们可以看出来ArticleFragment中只有UI初始化,发请网络请求action以及数据观察更新UI,列表展示用了TRecyclerView面向holder开发高复用,多类型的刷新库,从此只关心你的列表的Item展示。还通过泛型除去了MVP中通过接口传递信息的大量代码,从此see you Mass implementation of interfaces。

4-2:ArticleViewModel

/*** @author:tqzhang on 18/7/26 16:15*/public class ArticleViewModel extends BaseViewModel<ArticleRepository> {     private MutableLiveData<ArticleVo> mArticleData;     public ArticleViewModel(@NonNull Application application) {          super(application);     }     public LiveData<ArticleVo> getArticleList() {          if (mArticleData == null) {              mArticleData = new MutableLiveData<>();          }         return mArticleData;     }    public void getArticleList(String lectureLevel1, String lastId) {          mRepository.loadArticleRemList(new CallBack<ArticleVo>() {        @Override        public void onNext(ArticleVo articleObject) {            mArticleData.postValue(articleObject);        }        @Override        public void onError(String e) {        }                }}

ArticleViewModel中持有可被观察的数据持有者LiveData和真正发起网络请求动作,在接收到服务端返回的数据通过mArticleData.postValue(articleObject)方式通知注册的Observer进行数据的刷新,此处需注意的是,setValue方法只能在主线程中调用,而postValue可以在任何线程中调用,如果是在后台子线程中更新LiveData的值,必须调用postValue。

4-3:ArticleRepository

/** * @author:tqzhang on 18/7/28 13:00 */ public class ArticleRepository extends BaseRepository {      public void loadArticleRemList(final CallBack<ArticleVo> listener) {          addSubscribe(apiService.getArticleRemList()                .compose(RxSchedulers.io_main())                .subscribe(new RxSubscriber<ArticleVo>() {                    @Override                    public void onSuccess(ArticleVo articleObject) {                        listener.onNext(articleObject);                    }                    @Override                    public void onFailure(String msg) {                        listener.onError(msg);                    }            }));}

最后我们的ArticleRepository中就提供不含任何杂质的纯净的数据,此处只提供了网络层的数据,在实际应用中可拆分类loacl data和remote data,可根据实际项目需求自行处理。

至此一个简单的业务代码就完成了,是驴子是马,拉出来溜溜就知道,实践出真知,效果图奉上:

之前有码友们提到多个网络请求如何处理,如果同一页面有多个网络请求操作,其实多网络请求可以合并处理,统一返回结果然后组装数据返回,同时也有利于控制空页面的显示逻辑。

市面上各种各样开发架构,萝卜青菜各有所爱,没有最好的架构,只有最适合自己的,大家在选型开发架构时理应多多斟酌,当前很火的架构并不一定适合自己,结合自身项目进行舍取。

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

推荐阅读更多精彩内容