ViewModel的目的:存储数据,以注重生命周期的方式管理界面的相关数据
viewModel的特性:1,数据持久化,不依赖于Activity的生命周期,有自己独立的生命周期
2,异步回调不会造成内存泄漏
3,隔离Model层与View层
4,Fragments间共享数据
我们看一段代码
package com.example.testviewmodel230430;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
public class MainActivity extends AppCompatActivity {
private Object str = new Object();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
@Override
protected void onDestroy() {
super.onDestroy();
//如果屏幕旋转后hashcode值不一样,就证明我们重新创建了一个Activity,
//
Log.i("zhang_xin","Activity的hashCode值:"+this.hashCode());
Log.i("zhang_xin","Object的hashCode值:"+str.hashCode());
}
}
看下打印结果
zhang_xin com.example.testviewmodel230430 I Activity的hashCode值:161101531
zhang_xin com.example.testviewmodel230430 I Object的hashCode值:73226620
zhang_xin com.example.testviewmodel230430 I Activity的hashCode值:188408498
zhang_xin com.example.testviewmodel230430 I Object的hashCode值:59366924
Activity不一样了,它的成员变量也不一样了。
看一下viewModel的用法
布局文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:id="@+id/tvShow"
android:layout_width="match_parent"
android:layout_height="150dp"
android:text="数字"
android:gravity="center"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/tvClicked"
android:layout_width="match_parent"
android:layout_height="150dp"
android:gravity="center"
android:text="点击我"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</LinearLayout>
Activity
package com.example.testviewmodel230430;
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.ViewModelProvider;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
private User user = new User("张欣",18);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//这么写是错误的,这样属于重新创建了一个MainViewModel,该ViewModel是依赖于Activity的,
//这就失去了使用ViewModel的意义。
// MainViewModel mainViewModel = new MainViewModel();
//这样的写法会对数据进行特殊的处理
MainViewModel mainViewModel = new ViewModelProvider(this).get(MainViewModel.class);
TextView textView = findViewById(R.id.tvShow);
findViewById(R.id.tvClicked).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mainViewModel.number++;
textView.setText(mainViewModel.number+"");
}
});
}
}
viewmodel
package com.example.testviewmodel230430;
import androidx.lifecycle.ViewModel;
public class MainViewModel extends ViewModel {
public int number;
}
点击下面的按钮,上面TextView显示的数字每次+1。我们旋转下屏幕
额,和我们预想的不一样啊,数字又恢复到了初始值。哪里的问题?
要加这段绿色的代码哦,现在你随便旋转屏幕,MainViewModel的number都不会初始化了。
疑问:为什么数据在旋转后会保存?
Activity实现了ViewModelStoreOwner接口,
MainViewModel mainViewModel = new ViewModelProvider(this).get(MainViewModel.class);
这句代码中的this,就是指ViewModelStoreOwner,ViewModelStoreOwner里面有一个HashMap,这个HashMap里面的键是全类名,值就是ViewModel。