例如,我创建了包含2个活动的项目:FirstActivity with android:launchMode =" singleTask" flag和SecondActivity。
首先,用户启动FirstActivity,然后在按钮单击时启动SecondActivity。从SecondActivity我们创建状态栏通知。
Intent intent = new Intent(SecondActivity.this, FirstActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.putExtra(FirstActivity.ARG, true);
NotificationCompat.Builder builder = new NotificationCompat.Builder(SecondActivity.this);
builder.setSmallIcon(R.drawable.ic_launcher)
.setAutoCancel(true)
.setContentTitle("title")
.setContentText("message");
PendingIntent notifyIntent = PendingIntent.getActivity(SecondActivity.this, 0, intent,PendingIntent.FLAG_CANCEL_CURRENT);
builder.setContentIntent(notifyIntent);
NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
notificationManager.notify(1000, builder.build());
当我点击此通知并且我的应用程序现在正在运行时,将执行FirstActivity#onNewIntent方法并且intent包含ARG extra = true。这是好的情况。
当我点击此通知并且我的应用程序现在没有运行时,将不会执行FirstActivity#onNewIntent方法,但是将执行FirstActivity#onCreate()而不会执行我的ARG标记
getIntent().getBooleanExtra(ARG, false)
此意图包含ARG extra = true。这也是正确的情况。
但是当我在情境2按下后退按钮并再次运行应用程序后尝试退出我的应用程序时,我收到了FirstActivity#onCreate()意图,其中ARG = true(我处理的旧额外值)。为什么我的旗帜是" true"重新打开应用程序后?
答案 0 :(得分:6)
我在寻找类似问题的解决方案时遇到了这个问题。虽然这个问题来自一年前,但我会写下我是如何解决它的,以防它可以帮助别人,因为我花了很多时间来解决我的问题。 我还将使用android:launchMode声明的活动称为“singleTask”。从服务发送通知,此通知具有PendingIntent和Intent以打开MyActivity。
我的解决方案是在意图中设置标志FLAG_ACTIVITY_NEW_TASK(http://developer.android.com/reference/android/content/Intent.html#FLAG_ACTIVITY_NEW_TASK)。 然后使用id 0和PendingIntent.FLAG_CANCEL_CURRENT创建Pending intent。
Intent _intent = new Intent(context, MyActivity.class);
_intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
_intent.putExtra("DATA", myData);
PendingIntent _pIntent = PendingIntent.getActivity(context, 0, notificationIntent, PendingIntent.FLAG_CANCEL_CURRENT);
每当单击通知时,都会在新任务中创建MyActivity实例并调用onCreate。 Intent包含正确的额外数据。 当我按下后退按钮并有一个新通知时,意图总是带来最新的额外数据。 如果活动的实例已经存在(例如,当我按下主屏幕按钮时),则调用onNewIntent方法并且意图带来正确的新额外数据。
http://developer.android.com/guide/topics/manifest/activity-element.html#lmode清楚解释了这一点。 我还尝试在PendingIntent.getActivity和PendingIntent.FLAG_UPDATE_CURRENT中使用不同的请求代码(arg 2),但我认为启动模式的定义是singleTask和Intent.FLAG_ACTIVITY_NEW_TASK。
我还有另一个案例,其中launchMode是“标准”。此处还会通过单击服务发送的通知来启动活动。 为了让已启动的活动接收到正确的额外数据,我设置了标志FLAG_ACTIVITY_SINGLE_TOP。
_intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_REORDER_TO_FRONT);
当活动的实例已存在时(例如,当我按下主屏幕按钮时),此标志使得onNewIntent方法被调用。再次使用正确的额外数据。否则也会使用正确的额外数据调用onCreate(例如,当我按下后退按钮时)。
每次都会使用新的请求代码创建Pending intent,但它也应该使用相同的请求代码。我认为秘密在于launchMode和意图标志设置。
PendingIntent pIntent = PendingIntent.getActivity(this, notificationRequestCode, _intent, PendingIntent.FLAG_CANCEL_CURRENT);