如何使用不同的意图启动活动时,如何防止活动的多个实例

bsb*_*ley 116 android activity-stack google-play back-stack

当我使用Android市场上的"打开"按钮启动时,我在应用程序中遇到了一个错误.似乎从市场上推出它会使用不同的意图,然后从手机的应用程序菜单中启动它.这导致启动相同活动的多个副本,这些副本彼此冲突.

例如,如果我的应用程序包含ABC活动,则上述问题可能会导致堆栈ABCA.

我尝试使用Intent所有活动来解决这个问题,但每当我点击HOME时,它都会产生不必要的副作用,即将活动堆栈清除为root.

示例: ABC - > HOME - > A当我需要的是ABC - > HOME - > ABC

在使用HOME时,是否有一种防止启动相同类型的多个活动而不重置根活动的好方法?

小智 180

将此添加到onCreate,你应该很高兴:

// Possible work around for market launches. See https://issuetracker.google.com/issues/36907463
// for more details. Essentially, the market launches the main activity on top of other activities.
// we never want this to happen. Instead, we check if we are the root and if not, we finish.
if (!isTaskRoot()) {
    final Intent intent = getIntent();
    if (intent.hasCategory(Intent.CATEGORY_LAUNCHER) && Intent.ACTION_MAIN.equals(intent.getAction())) {
        Log.w(LOG_TAG, "Main Activity is not the root.  Finishing Main Activity instead of launching.");
        finish();
        return;       
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 多年来我一直在努力解决这个问题,这是有效的解决方案,所以非常感谢你!我还要注意,这不仅仅是Android Market中的一个问题,而且还通过将应用程序上传到服务器或通过电子邮件将其发送到手机来加载应用程序,会导致此问题.所有这些都使用Package Installer安装应用程序,我相信bug存在于此.此外,如果不清楚,您只需要将此代码添加到onCreate方法中,即根活动是什么. (24认同)
  • 只要您通过Eclipse(或IntelliJ或其他IDE)启动它,就可以使用从Eclipse部署的调试版本来实现**.它与应用程序如何在设备上安装**无关.问题是由于应用程序**的启动方式**. (6认同)
  • @CarlosP如果正在创建的活动不是**任务的根活动,那么**必须**(根据定义)至少是其下的一个其他活动.如果此活动调用`finish()`,则用户将看到下面的活动.因此,您可以安全地假设应用程序的现有实例将被带到前台.如果不是这种情况,您将在单独的任务中拥有应用程序的多个实例,并且正在创建的活动将是其任务的根. (4认同)
  • 我觉得这很奇怪,这发生在部署到设备的签名应用程序中,而不是从Eclipse部署的调试版本.调试非常困难! (2认同)
  • 有谁知道这段代码是否能确保将应用程序的现有实例带到前台?或者它只是调用finish(); 让用户看不到任何事情发生了吗? (2认同)

gil*_*ilm 25

我只想解释它失败的原因,以及如何以编程方式重现此错误,以便将其合并到测试套件中:

  1. 当您通过Eclipse或Market App启动应用程序时,它会使用意图标志启动:FLAG_ACTIVITY_NEW_TASK.

  2. 通过启动器(home)启动时,它使用标志:FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_BROUGHT_TO_FRONT | FLAG_ACTIVITY_RESET_TASK_IF_NEEDED,并使用操作" MAIN "和类别" LAUNCHER ".

如果您想在测试用例中重现这一点,请使用以下步骤:

adb shell am start -f 0x10000000 -n com.testfairy.tests.regression.taskroot/.MainActivity 
Run Code Online (Sandbox Code Playgroud)

然后做任何事情来进行其他活动.为了我的目的,我只是放了一个按钮来启动另一个活动.然后,回到启动器(主页):

adb shell am start -W -c android.intent.category.HOME -a android.intent.action.MAIN
Run Code Online (Sandbox Code Playgroud)

并模拟通过启动器启动它:

adb shell am start -a "android.intent.action.MAIN" -c "android.intent.category.LAUNCHER" -f 0x10600000 -n com.testfairy.tests.regression.taskroot/.MainActivity
Run Code Online (Sandbox Code Playgroud)

如果您尚未合并isTaskRoot()解决方法,则会重现该问题.我们在自动测试中使用它来确保此错误不再发生.

希望这可以帮助!


Eri*_*ine 7

您是否尝试过singleTop启动模式?

以下是http://developer.android.com/guide/topics/manifest/activity-element.html中的一些说明:

...还可以创建"singleTop"活动的新实例来处理新意图.然而,如果目标任务已在其堆栈顶部的活动的现有实例,该实例将接收新的意图(在onNewIntent()调用); 未创建新实例.在其他情况下 - 例如,如果"singleTop"活动的现有实例是目标任务,而不是在堆栈的顶部,或者如果它是在堆栈的顶部,而不是在目标任务 - 将创建新实例并将其推送到堆栈上.

  • 我想到了这一点,但如果活动不在堆栈顶部怎么办?例如,似乎singleTop会阻止AA,但不会阻止ABA. (2认同)