Android---Jetpack之ViewModel

ViewModel 的诞生

解决以下几个问题:

  \bullet 瞬态数据丢失

  \bullet 异步调用的内存泄漏

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

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

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

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

ViewModel 的作用

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

扫描二维码关注公众号,回复: 14791789 查看本文章

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

ViewModel 的简单应用

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

1. activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="0"
        android:textSize="24sp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintVertical_bias="0.247" />

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="+"
        android:textSize="34sp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        android:onClick="plusNumber"/>

</androidx.constraintlayout.widget.ConstraintLayout>

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 上的数据仍然存在。如下图:

注意

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

public class MyViewModel extends AndroidViewModel

猜你喜欢

转载自blog.csdn.net/qq_44950283/article/details/129793290