阅读 349

Android之MVVM架构指南(五):ViewModel

ViewModel 是Android 架构组件中负责管理UI相关数据与逻辑的,它的功能定义与MVP架构中的Persenter十分相似,配合其他组件使用增加许多方便开发的功能。

基本使用

定义一个ViewModel只需要继承ViewModel抽象类即可:

public class MainViewModel extends ViewModel {
   
}
复制代码

然后我们就可以在activity/fragment中实例化它:

MainViewModel viewModel = ViewModelProviders.of(this).get(MainViewModel.class);
复制代码

注意:如果想要正常使用ViewModel的全部功能,不能直接new出ViewModel的实例,而是使用ViewModelProvider类创建,上面的ViewModelProviders其实就是对ViewModelProvider的包装类。

正是因为通过ViewModelProvider创建ViewModel的方式,ViewModel才具备了一些比较方便实用的功能。

生命周期

因为创建ViewModel时传入了activity/fragment对象实例(ViewModelProviders.of(this)),所以ViewModel可以感知宿主的生命周期。

当宿主onDestroy()时候ViewModel便会自行销毁掉,除此之前,当屏幕旋转的时候,Activity会被recreate,Activity会经过几个生命周期方法,但是这个时候ViewModel还是之前的对象,并没有被重新创建。

共享数据

public class MainViewModel extends ViewModel{

}

public class Fragment1 extends Fragment{
    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        MainViewModel ViewModel = ViewModelProviders.of(getActivity()).get(MainViewModel.class);
    }
}

public class Fragment2 extends Fragment{
    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        MainViewModel ViewModel = ViewModelProviders.of(getActivity()).get(MainViewModel.class);
    }
}
复制代码

上面的例子中,如果fragment1和fragment2的宿主activity是同一个的话,那么它们创建的ViewModel是同一个,ViewModel中的数据二者可以共享。

更新UI

ViewModel配合LiveData使用可以做到在不持有activity任何引用的情况下更新UI数据,这是整个组件架构中最有亮点以及最有特色的地方。

public class MainViewModel extends ViewModel{
    MutableLiveData<String> data = new MutableLiveData<>();

    public LiveData<String> getData(){
        return data;
    }

    public void loadData(){
        // 模拟一个数据请求
        data.setValue("test");
    }

}

public class Fragment1 extends Fragment{
    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        MainViewModel ViewModel = ViewModelProviders.of(this).get(MainViewModel.class);
        ViewModel.getData().observe(this,new Observer(){
            @Override
            public void onChanged(String s){
                textView.setText(s);
            }
        });
    }
}
复制代码

在上面的例子中,textView会根据数据源的变化更新自己,而ViewModel中不需要考虑UI更新的问题,只需要关注数据的变化即可。

注意事项

ViewModel看似使用简单,但是在实际使用中会有一些点需要关注的。

1. 数据源的初始化

ViewModel本质上就是管理各种LiveData数据源的,但一定要注意的是在外界getData()前一定要初始化数据源,因为外界获得数据源后一般会直接注册观察者,如果这时候数据源没有初始化就会空指针异常。

2. 数据源的更新

上面的例子中loadData()方法更新数据源是直接用当前data对象更新的,但是在实际使用中每次请求数据后极有可能会发生数据请求模块返回一个新的LiveData实例,这种情况不能直接将当前的data重新赋值:

data = newData;
复制代码

这样写的话外界设置的观察者就会失效了。