尝试保存嵌套片段状态时出现异常[关键android:target_state不再存在片段]

时间:2017-01-19 12:51:10

标签: java android android-fragments exception android-nested-fragment

为了专注于这个问题,我将简化以下情况 - 我有一个活动 A 和一个片段 F ,它正在添加另一个片段即可。每个的简化代码是

活动A

@Override
protected void onCreate(Bundle savedInstanceState) {
    // do some stuff
    FragmentManager fm = getSupportFragmentManager();
    F f = new F();
    fm.beginTransaction()
            .add(R.id.content, f)
            .commit();
}

片段F

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    // do some stuff
    FragmentManager fm = getChildFragmentManager();
    FragmentTransaction transaction = fm.beginTransaction();
    ChildFragment childFragment = new ChildFragment();
    childFragment.setTargetFragment(this, 1);
    transaction.add(R.id.f, childFragment);
    transaction.commit();

    return view;
}

子片段的代码与问题无关,所以我不会发布它。

使用此代码,在我集成Firebase并开始获取以下崩溃报告之前,所有内容似乎都能正常运行

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.test.test/com.test.test.A}: java.lang.IllegalStateException: Fragment no longer exists for key android:target_state: index 1
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2377)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2429)
    at android.app.ActivityThread.access$800(ActivityThread.java:151)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1342)
    at android.os.Handler.dispatchMessage(Handler.java:110)
    at android.os.Looper.loop(Looper.java:193)
    at android.app.ActivityThread.main(ActivityThread.java:5341)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:515)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:825)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:641)
    at dalvik.system.NativeStart.main(Native Method)

起初我无法重现异常,但经过一段时间的测试后,我发现如果开发人员选项不要保持活动,则几乎每次我将活动带到后台时都会发生恢复它。我认为在正常情况下,当活动被置于后台并且应用程序被销毁时会发生。

在做了一些研究之后,我得出结论,崩溃的实际原因是片段F被设置为它的孩子的目标片段。我可以确认,如果我没有设置目标片段,则不会发生崩溃。

我不是绝对肯定,但似乎崩溃的原因是Child Fragment及其目标片段在不同的FragmentManagers中。所以我尝试的第一件事就是将所有片段放在activity的片段管理器中。

片段F

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    // do some stuff
    // I do not want to use private fragment manager but rather use the activity's
    // FragmentManager fm = getChildFragmentManager();
    FragmentManager fm = getFragmentManager();
    // do the other stuff
}

这解决了这个问题。但导致另一个。当我从活动中删除片段(我想用另一个片段替换它)。子片段无法保存其状态,因为它具有对其父片段的引用,该片段将从管理器中删除。

Process: com.test.test, PID: 11047 java.lang.IllegalStateException: Failure saving state: ChildFragment{423c10f0 #1 id=0x7f0b0058} has target not in fragment manager: F{423c0f88}
    at android.support.v4.app.FragmentManagerImpl.saveAllState(FragmentManager.java:2618)
    at android.support.v4.app.FragmentController.saveAllState(FragmentController.java:134)
    at android.support.v4.app.FragmentActivity.onSaveInstanceState(FragmentActivity.java:571)
    at android.support.v7.app.AppCompatActivity.onSaveInstanceState(AppCompatActivity.java:515)
    at android.app.Activity.performSaveInstanceState(Activity.java:1157)
    at android.app.Instrumentation.callActivityOnSaveInstanceState(Instrumentation.java:1229)

我可以尝试更深入地移除子片段,当其父片段被移除但我觉得这不是正确的方法,毕竟我认为正确的方法是使用getChildFragmentManager() 。

非常感谢有关该主题的任何帮助,建议和指南。

1 个答案:

答案 0 :(得分:3)

我已经改变了你的代码,它对我有用, 像这样更改ActivityA,

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

        FragmentManager fm = getSupportFragmentManager();
        if (fm.findFragmentById(R.id.content) == null) {
            F f = new F();
            fm.beginTransaction()
                    .add(R.id.content, f)
                    .commit();
        }
    }

您可以将之前的实现(如setTargetFragment)用于ChildFragment片段。这解决了您在问题中提到的第一个例外。