Android Jetpack:LiveData与ViewModel的使用教程

介绍

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方法。

获取Context上下文

使用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的使用》

你可能感兴趣的:(Jetpack,android,jetpack,android,java)