Why is onDestroy always called when returning to parent activity?

点点圈 提交于 2019-11-27 14:46:09

问题


I have a very simple app based on the Building Your First App tutorial. There are two activities: MainActivity invokes DisplayMessageActivity through startActivity().

When entering DisplayMessageActivity, I see:

MainActivity.onStop()

as expected, but when I press the back button to return to the parent MainActivity, I get:

MainActivity.onDestroy()
MainActivity.onCreate(null)
MainActivity.onStart()

The activity always gets destroyed for this very simple application. But according to the documentation (second bullet point), the typical behavior is for the activity to be stopped and restarted in such cases.

Also, onDestroy() does not happen when first starting the child activity, but only once back button is clicked.

Two questions:

  1. Is there a way to prevent parent from being destroyed in the common case?
  2. Why is null being passed to onCreate() here? This prevents me from preserving state through onSaveInstanceState().

Note that I've verified that Settings -> Developer Options -> Apps -> Don't keep activities is unchecked.

Edit:

Here is how the child activity is linked to parent:

    <activity
        android:name="com.example.helloworld.DisplayMessageActivity"
        android:label="@string/title_activity_display_message"
        android:parentActivityName="com.example.helloworld.MainActivity" >
        <meta-data
            android:name="android.support.PARENT_ACTIVITY"
            android:value="com.example.helloworld.MainActivity" />
    </activity>

Tracing through DisplayMessageActivity.onOptionsItemSelected(), I can see that it's calling Activity.onNavigateUp().


回答1:


Thanks to Greg Giacovelli's comments, I found the answer here. The solution was to set android:launchMode="singleTop" to the parent activity.

I still can't understand why such basic information is so unknown and hard to find!




回答2:


Adding launchMode in Manifest changes launch mode every time, even it is not launched by child activity. There are other ways to launch existed instance.

1.override onOptionsItemSelected(item: MenuItem)

override fun onOptionsItemSelected(item: MenuItem): Boolean {
   return when (item.itemId) {
        android.R.id.home -> {
            // Respond to the action bar's Up/Home button
            val upIntent: Intent? = NavUtils.getParentActivityIntent(this)
            when {
                upIntent == null -> throw IllegalStateException("No Parent Activity Intent")
                else -> {
                    //add launch flag here
                    upIntent.flags=Intent.FLAG_ACTIVITY_CLEAR_TOP
                    NavUtils.navigateUpTo(this, upIntent)
                }
            }
            true
        }
        else -> super.onOptionsItemSelected(item)
    }
}
  1. 2nd solution looks like a bug. Set "android:parentActivityName" to activity which is not existed in current task stack. Android will behavior like "BACK" button. It launches the top instance instead of launching parent activity. Document mentions "NavUtils.shouldUpRecreateTask" can recognize if parent activity existed in current task. But it doesn't work in my test. Navigate up with a new back stack


来源:https://stackoverflow.com/questions/20819019/why-is-ondestroy-always-called-when-returning-to-parent-activity

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!