3

通过单击启动器中的应用程序图标来启动活动,应该将活动带到前台,就像从历史中选择它一样。所以不应该存在 onCreate 调用。

但是,如果我们在通过单击通知启动 Activity 后尝试执行此操作,则启动器只会启动该 Activity 的另一个实例。

我必须添加哪些标志才能使启动器继续按预期工作(从后台恢复应用程序的确切状态)?

我将发布基本代码。

这将启动通知:

Intent resumeIntent = new Intent(this, MainActivity.class);
PendingIntent resumePendingIntent = PendingIntent.getActivity(this, 2,
        resumeIntent, PendingIntent.FLAG_UPDATE_CURRENT);

Notification resumeNotification = new Notification.Builder(this).setContentTitle(
        "Resume style")
        .setSmallIcon(R.drawable.ic_launcher)
        .setContentIntent(resumePendingIntent)
        .build();

NotificationManager notificationManager = (NotificationManager) getSystemService(Service.NOTIFICATION_SERVICE);
notificationManager.notify(1, launcherNotification);

这是清单活动的外观:

    <activity
        android:name="com.example.ihatenotifiicationsapp.MainActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>

这个 resumeIntent 将被 Android 自动添加到 FLAG_ACTIVITY_NEW_TASK。此标志将允许从后台恢复应用程序(如果存在)。

一切都很好,直到这里,但是,如果在您单击此通知并恢复应用程序后,您从启动器中单击应用程序,那么 Android 会启动另一个 MainActivity 实例。

这会破坏我的应用程序和后台堆栈(堆栈中有 2 个 MainActivity,对用户来说很奇怪)。

有趣的是,只有在您单击通知后才会发生这种情况(单击启动器行为以启动另一个实例)。

4

2 回答 2

1

如果你想要这种行为,你可以android:launchMode="singleTask"在你的标签中使用标志。activity如果当前有一个活动实例,这将防止操作系统启动任何其他实例。有关启动行为的更多信息,请参见 SDK Doku

我编辑了这个答案,对应于下面的 Emanuel Moecklin 评论。混合了lauchModes。

摘自多库:

系统在新任务的根部创建活动并将意图路由到它。但是,如果活动的实例已经存在,系统会通过调用其 onNewIntent() 方法将意图路由到现有实例,而不是创建一个新实例。

于 2013-10-22T12:54:25.343 回答
0

尝试

Intent resumeIntent = new Intent(this, MainActivity.class)
                         .setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP;

如果设置,并且正在启动的活动已经在当前任务中运行,那么不会启动该活动的新实例,而是关闭它上面的所有其他活动,并且此 Intent 将被传递到(现在顶部)作为新意图的旧活动。 http://developer.android.com/reference/android/content/Intent.html#FLAG_ACTIVITY_CLEAR_TOP

于 2013-10-22T13:01:40.167 回答