为什么调用 finish() 会导致 onCreate() 被调用并启动一个新的 Activity?

14

(是的,我已经查看了与此问题相关的现有问题。)

我正在从我的 Activity 的 Up 按钮侦听器中调用 finish()。但尽管 onDestroy() 被调用,但首先会调用 onPause(),然后令人惊讶的是会调用 onCreate(),这会引起实际问题。为什么通过调用 ScanningActivity 的 finish() 方法会启动一个新的 ScanningActivity

我正在记录所有生命周期函数的调用,并且顺序如下:

 inside onClick() Listener for up button.
         Inside onPause()
         Inside onCreate()  // this is what's hosing everything
         Inside onStart()
         Inside onResume()
         Inside onWindowFocusChanged()
         Inside onStop()
         Inside onDestroy()

在调用 finish() 后,为什么我会得到这个事件序列?以下是调用 finish()ScanningActivity 代码,注意,在 onCreate() 方法中分配的 onclick 监听器中:

@Override
public void onCreate(Bundle savedInstanceState)
{
    . . .

    button.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) 
        {
            Log.i("ScanningActivity", "inside onClick() -- Listener for up button being executed.");

            android.widget.LinearLayout layt = (android.widget.LinearLayout) vCustView;

            View vTemp = layt.findViewById(R.id.scanning_up);
            if (null != vTemp)
            {
                Button btnUp = (Button) vTemp;
                if (!btnUp.getText().equals("End"))  
                {
                    setResult(RESULT_CANCELED);

                    finish();
                    return;
                }
            }
         }
    . . .
}

根据Karakuri的要求,这里是清单:

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

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    //
    <uses-permission android:name="android.permission.SET_DEBUG_APP" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

        <uses-feature android:name="android.hardware.camera" />
        <uses-feature android:name="android.hardware.camera.autofocus" />

        <uses-permission android:name="android.permission.CAMERA" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ail_logo"
        android:label="@string/app_name"
        android:theme="@style/AssistTheme">
        <activity
            android:name=".HomeScreenActivity"
            android:label="@string/title_activity_home_screen"
            android:screenOrientation="portrait">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name=".GradeActivity"
            android:label="@string/title_activity_grade_assessment"
            android:screenOrientation="portrait"
            android:parentActivityName=".HomeScreenActivity"
            android:noHistory="true"
            />
        <activity
            android:name=".ScanningActivity"
            android:label="@string/title_activity_scanning"
            android:screenOrientation="portrait"
            android:parentActivityName=".GradeActivity"
            />
    </application>


</manifest>

...这里是调用代码,最初从GradeActivity内部启动ScanningActivity

if (bFinishedSuccessfully)
{
    try
    {
        Log.i("GradeActivity", "SOMEHOW THIS IS AGAIN STARTING THE ScanningActivity!!!");

        Intent intent = new Intent(this, ScanningActivity.class);
        intent.putExtra("SessionInfo", m_hmActivate);
        startActivity(intent);
    }
    catch (Exception ex)
    {
        Log.i("GradeActivity", "ex: " + ex.getMessage());
    }
}

更新: 我稍微修改了清单和调用代码,试图修复这个问题,但是没有成功。行为仍然与最初描述的完全相同。我已经用这些修改更新了问题。


3
你发布的代码信息不够。你能否展示你的AndroidManifest文件?此外,看起来这个活动是为了得到一个结果而启动的。也许你应该包含一些调用活动的代码? - Karakuri
在您发布的事件序列中,哪个“Activity”记录了每个方法? - Bryan
1
如果您计划调用 setResult(),则必须使用 startActivityForResult() 启动活动,而不是 startActivity() - natario
在启动活动时使用显式意图,无需添加try catch。 - Qamar
@Qamar -- 可能我是,但这并不是唯一明确调用启动ScanningActivity的地方。我在唯一发生这种情况的地方之前放置了一个日志声明,但它没有被调用。我已经接近可以再次尝试复制这个问题的地方了。 - Alyoshak
显示剩余12条评论
2个回答

1
我认为问题出在你的清单文件上,GradeActivity中添加了android:noHistory="true"参数。ScanningActivity中的setResult()将尝试创建父级Activity GradeActivity。如果你不需要在导航栈中保留GradeActivity,我认为你需要使用startActivityForResult从HomeScreenActivity启动ScanningActivity。

真的吗?嗯,我不能直接从HomeScreenActivity启动ScanningActivity,因为在前往ScanningActivity的途中需要在GradeActivity中完成一个关键步骤。但是如果在途中发生故障,用户必须返回到HomeScreenActivity。有理由相信使用noHistory会导致我的这种问题吗? - Alyoshak
哪个活动处理了来自ScanningActivity的结果?当ScanningActivity恢复时,GrateActivity将被销毁,因此,当您调用setResult时,android活动管理器将尝试向呼叫活动返回结果,以便创建GrateActivity的新实例。解决方法是使用GrateActivity的自定义结果从HomeScreenActivity开始,或者从GradeActivity中删除noHistory参数,并在需要时完成它。 - Anis BEN NSIR
我不确定它是如何找回HomeScreenActivity的,但它确实做到了,这正是我需要的。也许是因为它找不到GradeActivity的实例,所以它向后查找并找到调用GradeActivity(即HomeScreenActivity)的活动,并返回那里。??? - Alyoshak
好的,如果我理解你的应用程序导航(HomeSreenActivity--> GradeActivity --> ScanningActivity --> 返回到HomeScreenActivity),那么你需要从ScanningActivity获取结果吗?如果是这样,你在哪里处理它?你已经测试了从ScanningActivity返回的结果吗? - Anis BEN NSIR
不需要结果先生。当ScanningActivity结束时,我希望它回到HomeScreenActivity。 HomeScreenActivity不会查找结果。 - Alyoshak
显示剩余2条评论

1

在 Click listener 中使用 setResult(RESULT_OK); 替代 setResult(RESULT_CANCELED);

例如:

@Override
public void onCreate(Bundle savedInstanceState)
{
    . . .

    button.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) 
        {
            Log.i("ScanningActivity", "inside onClick() -- Listener for up button being executed.");

            android.widget.LinearLayout layt = (android.widget.LinearLayout) vCustView;

            View vTemp = layt.findViewById(R.id.scanning_up);
            if (null != vTemp)
            {
                Button btnUp = (Button) vTemp;
                if (!btnUp.getText().equals("End"))  
                {
                    setResult(RESULT_OK);

                    finish();
                    return;
                }
            }
         }
    . . .
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接