Activity的重建(recreate)与恢复
关于Activity的重建可以从以下三个方面去了解:
Activity会出现重建的三种情况
- 系统内存不足:Activity系统会杀死一个进程,在系统需要释放内存的时候,而位于进程内的Activity就会被销毁。在用户重新回到这个Activity的时候,这个Activity就会被重建
- Configuration Change(Activity横竖屏切换,进入多窗口模式):在横竖屏切换时,系统为了调整布局适应新的配置,在默认情况下会重建Activity
- 调用Activity的
onCreate()
方法:主题或语言切换时可以调用onCreate重建Activity使新的主题生效
Activity在重建时的状态变化
首先我们通过一段简单代码来看一下重建过程Activity的状态:
public class LifeCycleActivity extends AppCompatActivity {
private static final String TAG = LifeCycleActivity.class.getName();
private static final String KEY_TEST = "test";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_life_cycle);
TextView tvContent = findViewById(R.id.tv_content);
if (savedInstanceState == null) {
Log.i(TAG, " ---> first time onCreate");
} else {
Log.i(TAG, " ---> recreate");
String test = savedInstanceState.getString(KEY_TEST);
tvContent.setText(test);
}
findViewById(R.id.btn_start).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
recreate();
}
});
}
@Override
protected void onStart() {
super.onStart();
Log.i(TAG, " ---> onStart");
}
@Override
protected void onResume() {
super.onResume();
Log.i(TAG, " ---> onResume");
}
@Override
protected void onPause() {
super.onPause();
Log.i(TAG, " ---> onPause");
}
@Override
protected void onStop() {
super.onStop();
Log.i(TAG, " ---> onStop");
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.i(TAG, " ---> onDestory");
}
@Override
protected void onRestart() {
super.onRestart();
Log.i(TAG, " ---> onRestart");
}
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putString(KEY_TEST, "testing");
Log.i(TAG," ---> onSaveInstanceState");
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
String test = savedInstanceState.getString(KEY_TEST);
Log.i(TAG, " ---> onRestoreInstanceState");
}
}
App运行之后切换横竖屏,通过上面代码实验得到的结果是:
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> first time onCreate
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onStart
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onResume
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onPause
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onSaveInstanceState
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onStop
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onDestory
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> recreate
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onStart
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onRestoreInstanceState
I/cn.appblog.activitylifecycle.LifeCycleActivity: ---> onResume
在进入paused状态之后,会调用onSaveInstanceState
去保存需要的数据状态,在重建的时候可以在onCreate
和onRestoreInstanceState
中的参数去获取销毁前保存的数据。
Activity的状态恢复
在一些场景下你的Activity是被正常的销毁,当用户按下返回键,或者Activity内部调用finish()
方法,在用户和系统的观念中这个Activity的实例是已经不再需要了。在这些场景下Activity的销毁,不需要做其他额外的工作。但是如果系统销毁Activity是由于系统的限制(configuration和memory pressure),这种情况下为了保持用户用户体验,我们需要在用户回到Activity时恢复Activity的状态。这种情况下,尽管Activity的实例已经消失,但是系统会记住Activity的实例曾经存在过。如果用户尝试重新回到Activity的时候,系统会使用Activity在被销毁时保存的状态和数据去创建一个新的Activity实例。
系统用来恢复之前的状态而保存的数据叫做instance state,是存储在Bundle中的键值对集合。默认情况下,系统使用Bundle来存储layout中View的状态(例如在EditText中输入的信息),确保Activity重建时能恢复到之前的状态。你也可以用来存储一些你想要在重建时恢复的信息,例如一些成员变量的值和用户的进度信息。
ps:为了确保系统能够恢复你Activity中View的状态,每一个View都需要有一个独特的id(通过 android:id
属性定义)
但是Bundle不适合用来保存大量的数据,因为数据在保存时需要在主线程中序列化,会消耗系统内存。如果有比较多的数据需要保存时可以结合本地存储。
View的状态系统会自动保存我们不需要做额外的操作,但是有其他数据需要存储时可以重写onSaveInstanceState
方法:
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putString(KEY_TEST, "testing");
Log.i(TAG," ---> onSaveInstanceState");
}
数据在onCreate
或onRestoreInstanceState
中恢复:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_life_cycle);
TextView tvContent = findViewById(R.id.tv_content);
if (savedInstanceState == null) {
Log.i(TAG, " ---> first time onCreate");
} else {
Log.i(TAG, "recreate");
String test = savedInstanceState.getString(KEY_TEST);
tvContent.setText(test);
}
findViewById(R.id.btn_start).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
recreate();
Intent intent = new Intent(LifeCycleActivity.this, SecondActivity.class);
startActivity(intent);
}
});
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
String test = savedInstanceState.getString(KEY_TEST);
Log.i(TAG, "onRestoreInstanceState");
}
onSaveInstanceState
方法并不是每次都会在onPause之后被调用,如果Activity是被明确的close(用户点击返回按钮)或者调用finish时,onSaveInstanceState
不会被调用。
如果需要持久化数据,如保存到preference或database,最佳选择是Activity在前台的时候去保存,如果没有这样的机会可以在onStop方法中去保存。
版权声明:
作者:Joe.Ye
链接:https://www.appblog.cn/index.php/2023/02/26/recreate-and-restore-activity/
来源:APP全栈技术分享
文章版权归作者所有,未经允许请勿转载。
共有 0 条评论