Android---Jetpack之ViewModel

ViewModel 的诞生

解决以下几个问题:

  \bullet 瞬态数据丢失

  \bullet 异步调用的内存泄漏

当我们取以异步操作区网络请求时,而我们又在网络数据返回前点击了“返回按钮”,此时Activity 已经销毁了,但是网络请求的这个对象还在请求,且一直占据在内存里面。而 Activity 已经销毁了,就在也拿不到 请求网络的这个对象了,这就是内存泄漏了。

内存泄漏:一个对象我们已经引用(使用)不到它了,但它又占有着内存。GC 以为该对象还能够使用,就没有回收它。当这种情况过多,内存就会不断被占用,就会导致手机卡顿。

Android---Jetpack之ViewModel_第1张图片

 

  \bullet 类膨胀提高维护难度和测试难度

 在 Activity 里面写的代码越来越多,导致维护难度和测试难度不断提高。

 

ViewModel 的作用

  \bullet 它是介于 View(视图)和 Model(数据模型)之间的桥梁

  \bullet 使视图和数据能够分离也能保持通信

ViewModel 的简单应用

屏幕旋转后用户操作的数据仍然存在,即保存瞬时态数据。

 

1. activity_main.xml




    

    

2. MyViewModel.java

MyViewModel 继承自 ViewModle,里面有一个数据变量 number。

package com.example.viewmodel;

import androidx.lifecycle.ViewModel;

public class MyViewModel extends ViewModel {
    public int number;
}

3. MainActivity.java

通过 new ViewModelProvider.get() 方法来获取到自定义的 MyViewModel 对象。

package com.example.viewmodel;

import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.ViewModelProvider;

import android.os.Bundle;
import android.view.View;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {

    private TextView textView;
    private MyViewModel viewModel;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        textView = findViewById(R.id.textView);
        // 获取到 MyViewModel
        viewModel = new ViewModelProvider(this,
                new ViewModelProvider.AndroidViewModelFactory(getApplication())).get(MyViewModel.class);
        
        textView.setText(String.valueOf(viewModel.number));
    }

    /**
     * 点击 button 时,textView 的内容随着 viewModel.number 的改变而改变
     */
    public void plusNumber(View view) {
        textView.setText(String.valueOf(++viewModel.number));
    }
}

ViewModel 的生命周期独立于配置变化,不管 Activity 到了生命周期的那个阶段,ViewModel 上的数据它都存在且可以访问,即便 Activity 销毁了 ViewModel 上的数据仍然存在。如下图:

Android---Jetpack之ViewModel_第2张图片

注意

不要向 ViewModel 里传入 Context,会导致内存泄漏。如果要使用 Contetxt,那就使用 AndroidViewModel 中的 Application 来代替 Context。

public class MyViewModel extends AndroidViewModel

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