【问题标题】:Why onDestroy inActivity A not being called after a call to finish() in ActivityB为什么在ActivityB中调用finish()后没有调用onDestroy inActivity A
【发布时间】:2015-04-29 11:08:24
【问题描述】:

下一个过程很容易理解和重现,但会导致错误:

  • activityA 在其onCreate() 方法中启动一个activityB
  • activityB 已创建,我在其 onResume() 方法中调用 finish()
  • activityB onDestroy() 被调用
  • activityA onResume() 被调用
  • 在活动 A 中,我单击菜单按钮调用 finish() - 或按返回键。
  • activityA 已被删除,但 onDestroy() 未被调用且 A 仍然存在(adb shell dumpsys 'myPackageName' 表示存在的活动过多)

代码

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="gleroy.com.algo">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">
        <activity
            android:name=".activity.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>


        <activity
            android:name="gleroy.com.algo.activity.FakeA"
            android:label="@string/app_name"></activity>

        <activity
            android:name="gleroy.com.algo.activity.FakeB"
            android:label="@string/app_name"></activity>
    </application>
</manifest>

活动 A:

public class FakeA extends Activity {

    private final static String TAG = FakeA.class.getCanonicalName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        Log.d(TAG, "onCreate, taskId :" + getTaskId());
        super.onCreate(savedInstanceState);

        Intent intent = new Intent(FakeA.this, FakeB.class);
        startActivity(intent);
    }

    @Override
    protected void onResume() {
        Log.d(TAG, "onResume");
        super.onResume();
    }


    @Override
    protected void onDestroy() {
        Log.d(TAG, "onDestroy");
        super.onDestroy();
    }

    @TargetApi(Build.VERSION_CODES.HONEYCOMB)
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        super.onCreateOptionsMenu(menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {

        switch (item.getItemId()) {
            case R.id.stop_session_menu_item:
                /* stop and quit */
                finish();
                return false;
        }

        return super.onOptionsItemSelected(item);
    }
}

活动 B:

public class FakeB extends Activity {

    private final static String TAG = FakeB.class.getCanonicalName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        Log.d(TAG, "onCreate, taskId :"+getTaskId());
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void onResume() {
        super.onResume();
        Log.d(TAG, "onResume, isFinishing :" + isFinishing());
        finish();
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        Log.d(TAG, "onDestroy");
    }
}

Activity A 从 MainActivity 启动,其中包含一个简单的按钮:

 @Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            Intent intent = new Intent(MainActivity.this, FakeA.class);
            startActivity(intent);
        }
    });

所以我知道我们不能确定 onDestroy() 是否会被调用,但我的 ActivityA 显然在泄漏。

我还观察到,如果我在 ActivityA 中使用 TimerTimerTask 来延迟 startActivity 或在 ActivityB 中延迟 finish(),那么我就不再有这个错误了。

这里是事件:

  • FakeA onCreate,taskId :154
  • FakeA onResume
  • FakeA onPause, isFinishing : false
  • FakeB onCreate, taskId :154
  • FakeB onResume, isFinishing :false
  • FakeA onResume
  • FakeB onDestroy
  • 调用完成或按返回键 : FakeA onPause, isFinishing : true

【问题讨论】:

  • 在 onOptionsItemSelected() 中返回 true 并检查
  • false 或 true 不会改变任何东西
  • 你在 A.onResume() 中调用 startActivity(B) 吗?那个奇怪的用户体验的目的是什么?
  • 您的代码是在 onCreate() 中启动 activityB,而不是在 onResume() 中。你是什​​么意思?
  • @CPUTerminator 他的意思是,他的AcitivityA启动了ActivityB的onResume()方法

标签: android android-activity


【解决方案1】:

代替finish() 试试finishAffinity()。 据我所知: finish() 只是破坏了当前的直播 ActivityfinishAffinity() 销毁所有活动的Activities

【讨论】:

    【解决方案2】:

    最好的解决方案是简单地检测启动 Activity FakeB 是否有用。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多