天天看点

Android T-MVVM深度解耦开发框架

前言

自从官方mvp的Sample出来后,闹得热火朝天的mvp,小码也未能幸免加入MVP大坑中,入坑近2年的MVP的架构终于要说拜拜了,最近由于公司项目相对稳定,做了一次较大的重构,原来的MVP架构切换到了T-MVVM。整个项目清爽了许多。

问题

MVP缺点:

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

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

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

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

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

如果项目业务不是很多或者业务相对简单,其实完全没有必要使用mvp,反而让项目变得更为复杂。

T-MVVM代码,如果有帮助记得star哦https://github.com/SelfZhangTQ/T-MVVM

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

ViewModel优点:

* 同步关联生命周期,

* 数据共享 ,

* 复用性强 ,

LiveData优点:

* 确保UI界面的数据状态,

* 没有内存泄漏,不会因为Activity的不可见导致Crash,

* 不用再人为的处理生命周期,

* 共享资源,

此架构未使用DataBinding原由:

* 数据绑定增加Bug调试难度,

* 复杂的页面,model也会很大,虽然使用方便了也很容易保证了数据的一致性,当时长期持有,不利于释放内存,

* 数据双向绑定不利于View重用,

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,在通过ViewModelProviders.of(fragment).get(modelClass))实例化ViewModel,

到此我们的基类基本编写完毕。

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

4-1:ArticleFragment

/**

* @author:zhangtianqiu 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);

layoutManager=new new StaggeredGridLayoutManager(1, StaggeredGridLayoutManager.VERTICAL);

//获取网络数据

getRemoteData();

}

//初始化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,欢迎star哦面向holder开发高复用,多类型的刷新库,从此只关心你的列表的Item展示。

通过泛型除去了MVP中通过接口传递信息的大量代码,

从此see you Mass implementation of interfaces。

4-1: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,可根据项目需求自行处理。

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

Android T-MVVM深度解耦开发框架

项目地址github地址:https://github.com/SelfZhangTQ/T-MVVM,欢迎大家交流,star。

原文发布时间为:2018-09-11

本文作者:zhangtianqiu

本文来自云栖社区合作伙伴“

安卓巴士Android开发者门户

”,了解相关信息可以关注“

”。

继续阅读