Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Android lifecycle weirdness with singleTop and foreground service?

Here's the scenario:

  • Start Activity A
  • Activity A starts service S
  • Service S runs in foreground mode and shows up a notification which when pressed takes the user to Activity B (which has launchMode="singleTop")
  • Activity B shows up
  • Press HOME
  • Go into DDMS and kill your application process to simulate that your app died (press red STOP button)
  • Android will say "Rescheduling crashed service in 5000ms" (sometimes longer)
  • Service S restarts and notification is shown.
  • Press the notification icon when the service restarts...

...at this time, Android will recover both Activities A and B due to the fact the process ended unexpectedly. But despite the fact Activity B is singleTop android will spawn it AGAIN because the user clicked on the notification. This results into having A -> B -> B on the activity stack. Pressing back will take you again onto the first recovered instance of Activity B.

Can someone from the Android team clarify what is happening behind the scenes and how to avoid this? What is the best way to simulate that Android killed the process due to low memory? Is pressing STOP from DDMS good enough or an edge case and this should never happen under normal circumstances?

What is the difference between 'Force Stop' from Settings --> Applications versus STOP from DDMS?

Thanks in advance!

like image 504
dnkoutso Avatar asked Dec 01 '11 04:12

dnkoutso


2 Answers

This behavior should not change based on whether or not the process is killed. The activity manager first looks through the server-side stack to decide what to do, and once the stack has been adjusted appropriately it resumes whatever is now at the top of the stack.

Check your app to make sure it is not calling startActivity() when re-initializing or doing something else like that. Look in the log to see what activities are being started and the intents being used. Use "adb shell dumpsys activity" to see what the current activity stack looks like. Maybe you have cleared the task affinity, so the second activity B is being started in its own task (in which case singleTop would have no impact)?

Also it is really hard to help people if you don't include useful details about what you are doing. The relevant log statements at the different steps, the state of the activity stack shown by "adb shell dumpsys activity", etc.

like image 171
hackbod Avatar answered Oct 13 '22 04:10

hackbod


I am not too sure about the detailed differences between STOP in DDMS and Force Stop, but I am pretty sure DDMS bypasses some internal Android functionality that Force Stop would perform since I do not remember Android ever reinitializing my activities when i did a Force Stop. If this is true, then I am suspecting that what you are seeing is the result of two different tasks started by Android: one for the old, killed activities and another for when the killed service restarts. You can test this theory by setting the flag to "singleTask" and checking if the same behaviour occurs. Hope this helps.

like image 25
ACrazyOldMan Avatar answered Oct 13 '22 05:10

ACrazyOldMan