LiveData是Android Jetpack的一个组件,它具有监听生命周期的功能,保证在生命周期的活跃状态时及时通知观察者数据更新。使用LiveData并不需要太担心内存泄漏问题,因为它在创建时会绑定LifeCycle(例如Activity),当LifeCycle销毁时,LiveData便会自动与其解除绑定。
ViewModel则是MVVM模式里的概念,从名字也可以看出它是View(视图)和Model(模型)通信的桥梁。ViewModel的作用和MVP模式里的Presenter非常相似,但ViewModel却很好地解决了MVP接口过多的问题。
首先定义一个ViewModel类,功能很简单。LiveData本身是一个抽象类,MutableLiveData是它的实现类。
有一点需要注意,ViewModel是通过反射来创建的,所以它的继承类必须有无参构造方法。
public class CountViewModel extends ViewModel {
private MutableLiveData<Integer> mCountData;
public MutableLiveData<Integer> getCount(){
if(mCountData == null){
mCountData = new MutableLiveData<>();
}
return mCountData;
}
public void setNewCount(int newCount){
mCountData.setValue(newCount);
}
}
接下来写Activity的代码。Activity成为LiveData的观察者,因此LiveData能够监听到Activity的生命周期,Activity对象被销毁时,LiveData便能及时解除绑定。
public class MainActivity extends AppCompatActivity {
private CountViewModel mCountViewModel;
private TextView mTextView;
private Button mButton;
private int mCount;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = findViewById(R.id.text);
mButton = findViewById(R.id.button);
//通过CountViewModel.class来创建实例对象
mCountViewModel = new ViewModelProvider(this).get(CountViewModel.class);
//Activity成为此LiveData的观察者,LiveData能够监听到Activity的生命周期
mCountViewModel.getCountLiveData().observe(this, new Observer<Integer>() {
@Override
public void onChanged(Integer newCount) {
mTextView.setText("" + newCount);
}
});
mButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mCountViewModel.setNewCount(mCount++);
}
});
}
}
在CountViewModel类的setNewCount方法中,LiveData调用setValue方法,此时和它绑定的Activity便能得到数据变更的通知。于是就调用了上面的onChanged方法。
//CountViewModel.java
public void setNewCount(int newCount){
mCountData.setValue(newCount);
}
我们在原来的CountViewModel类上做一些修改,实现数据类型转换。mCountStrData保存字符串数据,是由mCountData转换而来的,
public class CountViewModel extends ViewModel {
private MutableLiveData<Integer> mCountData;
private LiveData<String> mCountStrData;
public MutableLiveData<Integer> getCountLiveData(){
if(mCountData == null){
mCountData = new MutableLiveData<>();
}
return mCountData;
}
public LiveData<String> getCountStrLiveData(){
if(mCountStrData == null){
//mCountStrData与mCountData绑定
mCountStrData = Transformations.map(getCountLiveData(), new Function<Integer, String>() {
@Override
public String apply(Integer input) {
String result = "count:" + input;
return result;
}
});
}
return mCountStrData;
}
public void setNewCount(int newCount){
mCountData.setValue(newCount);
}
}
把Activity和mCountStrData绑定。
public class MainActivity extends AppCompatActivity {
private CountViewModel mCountViewModel;
private TextView mTextView;
private Button mButton;
private int mCount;
@Override
protected void onCreate(Bundle savedInstanceState) {
...
mCountViewModel.getCountStrLiveData().observe(this, new Observer<String>() {
@Override
public void onChanged(String s) {
Log.d("MainActivityTag", s);
}
});
}
}
点击按钮,mCountData数据被更改,而日志也会跟着打印。那是因为Transformations.map方法把mCountStrData和mCountData绑定在了一起。mCountData改变,mCountStrData也会跟着改变,于是它的观察者也会跟着触发onChanged方法。
使用ViewModel的时候有两个规定。
第一是LiveData应该放在ViewModel里,而尽量避免放在Activity或Fragment。因为LiveData把数据处理的代码转移到了ViewModel里,Activity主要负责显示数据(即充当视图的角色)。
第二是ViewModel不应该传入或引用Activity以及任何控件。因为一旦Activity销毁,而ViewModel又持有它的引用,就会导致内存泄漏。
那么ViewModel要怎么获取Context呢?其实还有一个类叫AndroidViewModel,继承它就行了。AndroidViewModel在创建时会绑定Application的Context,从这里也能看出来ViewModel往往比Activity存活更久。
本篇文章讲述了LiveData和ViewModel的使用方法。
《Android开发者文档》
《Android之ViewModel的使用》