【问题标题】:java.lang.IllegalStateException: Fragment already addedjava.lang.IllegalStateException:已添加片段
【发布时间】:2013-10-18 05:02:02
【问题描述】:

我在使用目标 SDK 4.3 编译和运行的 Android 应用时遇到问题。该应用程序有两个活动,一个也是启动器活动的 MainActivity 和一个 SecondActivity。两者都在使用片段。为了也支持旧设备,使用了支持库。

在以下场景中,会出现“IllegalStateException: Fragment already added”错误。

1) 启动应用程序 -> 显示 MainActivity
2) 使用 Intent
切换到 SecondActivity 3) 按主页按钮
4) 等待更长的时间(经过 24 小时测试)
5) 再次按下应用程序图标 -> 例外。如果时间更短,SecondActivity 会按预期显示。

我在 Fragment 处理期间阅读了很多 IllegalStateExceptions,但所有这些都指出了 replace() 方法的问题。在 Stacktrace 中,我自己的代码永远不会被调用。

在 Activies 的 onCreate() 方法中添加 Fragment:

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(..);
    ListFragment listFragment = this.getCaptureListFragment();
    FragmentTransaction tx = this.getSupportFragmentManager().beginTransaction();
    tx.add(R.id.MainFragmentContainer, listFragment, "list_fragment_tag");
    tx.commit(); 
}

private ListFragment getListFragment() {
    ListFragment listFragment = (ListFragment) this.getSupportFragmentManager().findFragmentByTag("list_fragment_tag");
    if (listFragment == null) {
        listFragment = new ListFragment();
    }
    return listFragment;
}


java.lang.RuntimeException: Unable to start activity    ComponentInfo{de.myexample.demo/de.myexample.demo.ui.SecondActivity}: java.lang.IllegalStateException: Fragment already added: ListFragment{42283f58 #0 id=0x7f060094 de.myexample.demo.ui.ListFragment}
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2211)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
    at android.app.ActivityThread.access$600(ActivityThread.java:141)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1256)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:137)
    at android.app.ActivityThread.main(ActivityThread.java:5103)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:525)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
    at dalvik.system.NativeStart.main(Native Method)
Caused by: java.lang.IllegalStateException: Fragment already added: ListFragment{42283f58 #0 id=0x7f060094 de.myexample.demo.ui.ListFragment}
    at android.support.v4.app.FragmentManagerImpl.addFragment(SourceFile:1175)
    at android.support.v4.app.BackStackRecord.run(SourceFile:616)
    at android.support.v4.app.FragmentManagerImpl.execPendingActions(SourceFile:1460)
    at android.support.v4.app.FragmentActivity.onStart(SourceFile:556)
    at android.app.Instrumentation.callActivityOnStart(Instrumentation.java:1171)
    at android.app.Activity.performStart(Activity.java:5143)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2184)
    ... 11 more
java.lang.IllegalStateException: Fragment already added: ListFragment{42283f58 #0 id=0x7f060094 de.myexample.demo.ui.ListFragment}
    at android.support.v4.app.FragmentManagerImpl.addFragment(SourceFile:1175)
    at android.support.v4.app.BackStackRecord.run(SourceFile:616)
    at android.support.v4.app.FragmentManagerImpl.execPendingActions(SourceFile:1460)
    at android.support.v4.app.FragmentActivity.onStart(SourceFile:556)
    at android.app.Instrumentation.callActivityOnStart(Instrumentation.java:1171)
    at android.app.Activity.performStart(Activity.java:5143)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2184)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
    at android.app.ActivityThread.access$600(ActivityThread.java:141)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1256)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:137)
    at android.app.ActivityThread.main(ActivityThread.java:5103)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:525)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
    at dalvik.system.NativeStart.main(Native Method)

【问题讨论】:

  • activity 的 setContentView 在哪里?
  • 尝试使用 tx.replace(R.id.MainFragmentContainer, listFragment);而不是使用 tx.add() 方法。阅读this。有关片段的更多信息
  • 谢谢@Raghunandan。我在帖子中添加了缺失的行。
  • @karannagpal:感谢您的链接,但我已经阅读了该手册。当然,我一定做错了什么,但我不确定是什么。我肯定会尝试替换方法。但我想了解为什么只有当应用程序在后台运行时间很长时才会出现该错误。我知道android会同时清理东西,但是Activity的重新创建似乎与正常生命周期不同。这对我来说很有趣
  • @karannagpal 替换肯定不起作用。在这种情况下,我自己的代码不会被调用。在应用程序进入后台之前如何添加片段并不重要

标签: java android android-fragments


【解决方案1】:

好的,我自己解决了。

我将所有片段放在 onPause() 中,并将状态存储在一些布尔值中。根据该布尔值,将片段放入 onResume() 中。无论 Activity 在后台运行多长时间,启动都是稳定的。

boolean addList = false;

@Override
protected void onResume() {
    FragmentTransaction tx = this.getSupportFragmentManager().beginTransaction();
    if (this.addList) {
        ListFragment list = this.getListFragment();
        tx.add(R.id.MainFragmentContainer, list, "list_fragment_tag");
    }

    tx.commit();
    super.onResume();

    this.addList = false;   

}

@Override
protected void onPause() {
    this.addList = this.getListFragment().isAdded();
    ...
    if (this.addList) {
        FragmentTransaction tx = this.getSupportFragmentManager().beginTransaction();
        tx.remove(this.getListFragment());
        tx.commit();
    }
    this.getSupportFragmentManager().executePendingTransactions();
    super.onPause();

}

也许这对有同样问题的人有帮助

【讨论】:

    【解决方案2】:

    为了重现这一点,可以在设置 -> 开发人员选项中激活“不保留活动”。然后暂停并恢复活动。

    这样您就不必等待 24 小时 :)

    【讨论】:

      【解决方案3】:

      片段管理器在退出时保存其状态。所以你不必再次添加片段。

      这样做:

      protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
          setContentView(..);
      
          if (savedInstanceState == null) {
              ListFragment listFragment = this.getCaptureListFragment();
              FragmentTransaction tx = this.getSupportFragmentManager().beginTransaction();
              tx.add(R.id.MainFragmentContainer, listFragment, "list_fragment_tag");
              tx.commit(); 
          }
      }
      

      【讨论】:

      • 感谢您的回答。它有所帮助,但并不完全。在 Android 4.1 上,它可以解决问题。但是在 4.4 KitKat 设备上,活动是空的,看不到任何片段。但看起来,这种方式比手动删除要好得多。也许,我可以在 if 内的较新设备上手动删除片段......我必须做更多测试。但是它们非常耗时,因为我必须至少等待 24 小时才能解决问题
      【解决方案4】:

      不需要创建新的布尔字段来检查片段的添加状态。在片段中也有一个方法。 只需使用它: myFragment.isAdded()

      【讨论】:

      • 布尔值不是用来存储被添加的片段的状态。它用于存储之前的状态(参见我的 onPause() 方法)。我认为你没有理解这个问题。
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-03-20
      • 1970-01-01
      • 1970-01-01
      • 2020-11-16
      • 2011-09-09
      相关资源
      最近更新 更多