为了专注于这个问题,我将简化以下情况 - 我有一个活动 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() 。
非常感谢有关该主题的任何帮助,建议和指南。
答案 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片段。这解决了您在问题中提到的第一个例外。