各种 Android 系统操作都会影响 fragment 的状态。为了确保用户的状态得到保存,Android 框架会自动保存和恢复 fragment 和返回堆栈。因此,您需要确保您的 fragment 中的所有数据也得到保存和恢复。
下表列出了会导致 fragment 丢失状态的操作,以及各种类型的状态是否会在发生这些更改后持续存在。表中提及的状态类型如下:
- 变量:fragment 中的局部变量。
- 视图状态:fragment 中一个或多个视图所拥有的任何数据。
- SavedState:此 fragment 实例固有的数据,这些数据应保存在
onSaveInstanceState()
中。 - NonConfig:从外部来源(如服务器或本地代码库)提取的数据,或提交后发送到服务器的由用户创建的数据。
通常,将变量视为与 SavedState 相同,但下表对这两者进行了区分,以说明各种操作对每种状态类型所产生的影响。
操作 | 变量 | 视图状态 | SavedState | NonConfig |
---|---|---|---|---|
添加到返回堆栈 | ✓ | ✓ | x | ✓ |
配置更改 | x | ✓ | ✓ | ✓ |
进程终止/重新创建 | x | ✓ | ✓ | ✓* |
被移除,不添加到返回堆栈 | x | x | x | x |
宿主已完成 | x | x | x | x |
* 可以使用 ViewModel 的已保存状态模块在进程终止后保留 NonConfig 状态。
表 1:各种 fragment 破坏性操作以及它们对不同状态类型所产生的影响。
我们来看一个具体的示例。假设有一个屏幕,该屏幕生成了一个随机字符串,将其显示在 TextView
中,并提供了一个选项,让用户可以在将该字符串发送给好友之前对其进行修改:
在本例中,假设用户按修改按钮后,应用会显示 EditText
视图,用户可以从中修改消息。如果用户点击取消,系统应清除 EditText
视图,并将其可见性设为 View.GONE
。这样的屏幕可能需要管理四块数据,以确保提供无缝的体验:
数据 | 类型 | 状态类型 | 说明 |
---|---|---|---|
seed |
Long |
NonConfig | 用于随机生成新善行的种子。创建 ViewModel 时生成。 |
randomGoodDeed |
String |
SavedState + 变量 | 首次创建 fragment 时生成。系统会保存 randomGoodDeed ,以确保即使在进程终止和重新创建后,用户也会看到相同的随机善行。 |
isEditing |
Boolean |
SavedState + 变量 | 当用户开始修改时,布尔标记设为 true 。系统会保存 isEditing ,以确保重新创建 fragment 后,屏幕的修改部分仍然可见。 |
修改后的文本 | Editable |
视图状态(由 EditText 拥有) |
EditText 视图中修改后的文本。EditText 视图会保存此文本,确保不会丢失用户正在进行的更改。 |
表 2:随机文本生成器应用必须管理的状态。
下面几部分介绍了如何正确管理数据在破坏性操作之后的状态。
视图状态
视图负责管理自己的状态。例如,当某个视图接受用户输入时,该视图负责保存和恢复该输入以处理配置更改。Android 框架提供的所有视图都有自己的 onSaveInstanceState()
和 onRestoreInstanceState()
实现,因此您不必管理 fragment 中的视图状态。
例如,在前面的场景中,修改后的字符串保存在 EditText
中。EditText
知道它当前显示的文本的值以及其他详细信息,如任何选定文本的开头和结尾。
视图需要一个 ID 才能保留其状态。此 ID 在 fragment 及其视图层次结构中必须是唯一的。没有 ID 的视图无法保留其状态。
<EditText android:id="@+id/good_deed_edit_text" android:layout_width="match_parent" android:layout_height="wrap_content" />
如表 1 所述,视图会保存其 ViewState
,并在所有不移除 fragment 或销毁宿主的操作之后恢复该状态。
SavedState
fragment 负责管理其正常运行所不可或缺的少量动态状态。您可以使用 Fragment.onSaveInstanceState(Bundle)
来保留易于序列化的数据。与 Activity.onSaveInstanceState(Bundle)
类似,您放置在捆绑包中的数据会在配置更改以及进程终止和重新创建后保留,并且在 fragment 的 onCreate(Bundle)
、onCreateView(LayoutInflater, ViewGroup, Bundle)
和 onViewCreated(View, Bundle)
方法中可用。
接着前面的示例来讲,randomGoodDeed
是向用户显示的善行,isEditing
是一个标记,用于确定 fragment 要显示还是隐藏 EditText
。这种已保存状态应使用 onSaveInstanceState(Bundle)
来持久保留,如以下示例所示:
Kotlin
override fun onSaveInstanceState(outState: Bundle) { super.onSaveInstanceState(outState) outState.putBoolean(IS_EDITING_KEY, isEditing) outState.putString(RANDOM_GOOD_DEED_KEY, randomGoodDeed) }
Java
@Override public void onSaveInstanceState(@NonNull Bundle outState) { super.onSaveInstanceState(outState); outState.putBoolean(IS_EDITING_KEY, isEditing); outState.putString(RANDOM_GOOD_DEED_KEY, randomGoodDeed); }
如需恢复 onCreate(Bundle)
中的状态,请从捆绑包中检索存储的���:
Kotlin
override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) isEditing = savedInstanceState?.getBoolean(IS_EDITING_KEY, false) randomGoodDeed = savedInstanceState?.getString(RANDOM_GOOD_DEED_KEY) ?: viewModel.generateRandomGoodDeed() }
Java
@Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); if (savedInstanceState != null) { isEditing = savedInstanceState.getBoolean(IS_EDITING_KEY, false); randomGoodDeed = savedInstanceState.getString(RANDOM_GOOD_DEED_KEY); } else { randomGoodDeed = viewModel.generateRandomGoodDeed(); } }
如表 1 所述,请注意,将 fragment 放置在返回堆栈上时,会保留变量。将变量视为已保存状态可确保它们在所有破坏性操作之后持续存在。
NonConfig
应将 NonConfig 数据放置在 fragment 之外,例如放置在 ViewModel
中。在前面的示例中,seed
(NonConfig 状态)是在 ViewModel
中生成的。维护其状态的逻辑由 ViewModel
拥有。
Kotlin
public class RandomGoodDeedViewModel : ViewModel() { private val seed = ... // Generate the seed private fun generateRandomGoodDeed(): String { val goodDeed = ... // Generate a random good deed using the seed return goodDeed } }
Java
public class RandomGoodDeedViewModel extends ViewModel { private Long seed = ... // Generate the seed private String generateRandomGoodDeed() { String goodDeed = ... // Generate a random good deed using the seed return goodDeed; } }
ViewModel
类本来就能让数据在发生屏幕旋转等配置更改后继续留存,并且在将 fragment 放置在返回堆栈上时会保留在内存中。进程终止和重新创建后,会重新创建 ViewModel
,并生成新的 seed
。将 SavedState
模块添加到 ViewModel
可让 ViewModel
在进程终止和重新创建后保留简单的状态。
其他资源
如需详细了解如何管理 fragment 状态,请参阅下面列出的其他资源。
Codelab
- 生命周期感知型组件 Codelab