导航和返回堆栈

NavController 保持一个“返回堆栈”,其中包含用户访问过的目标。当用户在您的应用中导航到屏幕时,NavController 会向返回堆栈添加和删除目标。

作为堆栈,返回堆栈是“后进先出”数据结构。NavController 因此将项目推送到堆栈顶部并从堆栈顶部弹出项目。

基本行为

以下是在考虑返回堆栈行为时应考虑的关键事实

  • 第一个目标:当用户打开应用时,NavController 会将第一个目标推送到返回堆栈的顶部。
  • 推送到堆栈:每个调用 NavController.navigate() 将给定目标推送到堆栈顶部。
  • 弹出顶部目标:点击向上返回分别调用 NavController.navigateUp()NavController.popBackStack() 方法。它们会将顶部目标从堆栈中弹出。有关向上返回之间区别的更多信息,请参阅导航原则页面。

返回

NavController.popBackStack() 方法尝试将当前目标从返回堆栈中弹出并导航到上一个目标。这实际上会将用户在导航历史记录中向后退一步。它返回一个布尔值,指示它是否成功返回到目标。

返回到特定目标

您还可以使用 popBackStack() 导航到特定目标。为此,请使用其重载之一。有几个允许您传入标识符,例如整数 id 或字符串 route。这些重载将用户带到与给定标识符关联的目标。至关重要的是,它们会弹出该目标上方堆栈上的所有内容。

这些重载还采用 inclusive 布尔值。它确定 NavController 是否也应在导航到指定目标后将其从返回堆栈中弹出。

考虑以下简短代码段作为示例

navController.popBackStack(R.id.destinationId, true)

这里 NavController 返回到具有整数 id destinationId 的目标。由于 inclusive 参数的值为 true,因此 NavController 还将给定目标从返回堆栈中弹出。

处理返回失败

popBackStack() 返回 false 时,后续对 NavController.getCurrentDestination() 的调用将返回 null。这意味着应用已将最后一个目标从返回堆栈中弹出。在这种情况下,用户只会看到一个空白屏幕。

这可能发生在以下情况下

  • popBackStack() 未从堆栈中弹出任何内容。
  • popBackStack() 已将目标从返回堆栈中弹出,并且堆栈现在为空。

要解决此问题,您必须导航到新目标或在您的活动上调用 finish() 以结束它。以下代码段演示了这一点

Kotlin

...

if (!navController.popBackStack()) {
    // Call finish() on your Activity
    finish()
}

Java

...

if (!navController.popBackStack()) {
    // Call finish() on your Activity
    finish();
}

向上返回到目标

在从一个目的地导航到另一个目的地时,要从返回栈中移除目的地,请向关联的 navigate() 函数调用添加 popUpTo() 参数。popUpTo() 指示导航库在调用 navigate() 的过程中从返回栈中移除一些目的地。参数值是返回栈上目的地的标识符。标识符可以是整数 id 或字符串 route

您可以包含 inclusive 参数的参数,其值为 true,以指示您在 popUpTo() 中指定的目的地也应该从返回栈中弹出。

要以编程方式实现此功能,请将 popUpTo() 作为 NavOptions 的一部分传递给 navigate(),并将 inclusive 设置为 true。这在 Compose 和 Views 中都有效。

弹出时保存状态

当您使用 popUpTo 导航到某个目的地时,您可以选择保存返回栈以及从返回栈中弹出的所有目的地的状态。然后,您可以在稍后导航到该目的地时恢复返回栈和目的地。这使您可以为给定目的地保留状态,并拥有 多个返回栈

要以编程方式执行此操作,请在将 popUpTo 添加到导航选项时指定 saveState = true

您也可以在导航选项中指定 restoreState = true 以自动恢复返回栈以及与目的地关联的状态。

例如

navController.navigate(
    route = route,
    navOptions =  navOptions {
        popUpTo<A>{ saveState = true }
        restoreState = true
    }
)

要在 XML 中启用保存和恢复状态,请分别在关联的 action 中将 popUpToSaveState 定义为 true,并将 restoreState 定义为 true

XML 示例

以下是在 XML 中使用 action 的 popUpTo 示例

<action
  android:id="@+id/action_a_to_b"
  app:destination="@id/b"
  app:popUpTo="@+id/a"
  app:popUpToInclusive="true"
  app:restoreState=”true”
  app:popUpToSaveState="true"/>

Compose 示例

以下是 Compose 中相同示例的完整示例

@Composable
fun MyAppNavHost(
    modifier: Modifier = Modifier,
    navController: NavHostController = rememberNavController(),
    startDestination: Any = A
) {
    NavHost(
        modifier = modifier,
        navController = navController,
        startDestination = startDestination
    ) {
        composable<A> {
            DestinationA(
                onNavigateToB = {
                // Pop everything up to, and including, the A destination off
                // the back stack, saving the back stack and the state of its
                // destinations.
                // Then restore any previous back stack state associated with
                // the B destination.
                // Finally navigate to the B destination.
                    navController.navigate(route = B) {
                        popUpTo<A> {
                            inclusive = true
                            saveState = true
                        }
                        restoreState = true
                    }
                },
            )
        }
        composable<B> { DestinationB(/* ... */) }
    }
}

@Composable
fun DestinationA(onNavigateToB: () -> Unit) {
    Button(onClick = onNavigateToB) {
        Text("Go to A")
    }
}

更细致地,您可以通过以下方式更改调用 NavController.navigate() 的方式

// Pop everything up to the destination_a destination off the back stack before
// navigating to the "destination_b" destination
navController.navigate("destination_b") {
    popUpTo("destination_a")
}

// Pop everything up to and including the "destination_a" destination off
// the back stack before navigating to the "destination_b" destination
navController.navigate("destination_b") {
    popUpTo("destination_a") { inclusive = true }
}

// Navigate to the "search” destination only if we’re not already on
// the "search" destination, avoiding multiple copies on the top of the
// back stack
navController.navigate("search") {
    launchSingleTop = true
}

有关将选项传递给 NavController.navigate() 的一般信息,请参阅 使用选项导航指南

使用操作弹出

在使用操作导航时,您可以选择从返回栈中弹出其他目的地。例如,如果您的应用具有初始登录流程,则在用户登录后,您应该将所有与登录相关的目的地从返回栈中弹出,以便“返回”按钮不会将用户带回登录流程。

其他阅读资料

有关更多信息,请阅读以下页面

  • 循环导航:了解如何在导航流程循环的情况下避免返回栈过载。
  • 对话框目的地:了解对话框目的地如何对您管理返回栈的方式提出独特考虑因素。