为什么 Android O 因 "does not belong to this FragmentManager!" 而失败

Why is Android O failing with "does not belong to this FragmentManager!"

我已将我的应用程序迁移到 Android O in Android Studio 3

运行 在 Android O 模拟器上,我所有的 dialogFragments 现在都失败了:-

java.lang.IllegalStateException: Fragment MyDialogFragment{43ccf50 #2 MyDialogFragment} declared target fragment SettingsFragment{ceed549 #0 id=0x7f0f0142 android:switcher:2131689794:0} that does not belong to this FragmentManager!
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1316)
at android.support.v4.app.FragmentManagerImpl.moveFragmentToExpectedState(FragmentManager.java:1624)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1689)
at android.support.v4.app.BackStackRecord.executeOps(BackStackRecord.java:794)
at android.support.v4.app.FragmentManagerImpl.executeOps(FragmentManager.java:2470)
at android.support.v4.app.FragmentManagerImpl.executeOpsTogether(FragmentManager.java:2260)
at android.support.v4.app.FragmentManagerImpl.optimizeAndExecuteOps(FragmentManager.java:2213)
at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:2122)
at android.support.v4.app.FragmentManagerImpl.run(FragmentManager.java:746)
at android.os.Handler.handleCallback(Handler.java:769)
at android.os.Handler.dispatchMessage(Handler.java:98)
at android.os.Looper.loop(Looper.java:164)
at android.app.ActivityThread.main(ActivityThread.java:6535)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)

我没有做任何代码更改。

Android O 以前工作的 DialogFragments 现在无法显示有什么变化?

Android Studio 3.0 Canary 1
Build #AI-171.4010489, built on May 15, 2017
JRE: 1.8.0_112-release-b736 x86_64
JVM: OpenJDK 64-Bit Server VM by JetBrains s.r.o
Mac OS X 10.11.6

    compileSdkVersion 'android-O'
    buildToolsVersion "26.0.0-rc2"

   AndroidManifest.xml
    defaultConfig {
        minSdkVersion 16
        targetSdkVersion 'O'
        }

compile 'com.android.support:appcompat-v7:26.0.0-beta1'
compile 'com.android.support:cardview-v7:26.0.0-beta1'
compile 'com.android.support:design:26.0.0-beta1'
    compile 'com.android.support:percent:26.0.0-beta1'

  dependencies {
        classpath 'com.android.tools.build:gradle:3.0.0-alpha1'

    }

我遇到了同样的问题,绝对是 android 错误。当您使用另一个片段作为目标显示片段时,就会发生这种情况。作为解决方法,您可以使用:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O)
    getActivity().getFragmentManager().beginTransaction().add(dialogFrag, "dialog").commit();
else
    getChildFragmentManager().beginTransaction().add(dialogFrag,"dialog").commit();

对我来说,这不仅是 Android O 的问题,也是旧版本的问题。 我测试的最旧版本是 API 16 级。

我正在使用以下代码实例化我的片段:

MyFragment myFragment = MyFragment.newInstance();
myFragment.setTargetFragment(ParentFragment.this, 0);
myFragment.show(getActivity().getSupportFragmentManager(), null);

其中 ParentFragment.this 是自定义 class 扩展 android.support.v4.app.FragmentMyFragment 也扩展此 class 并且是 [=15= 的子片段] 片段(因此得名)。

我认为我必须使用 SupportFragmentManager(getSupportFragmentManager() 方法),因为我使用的是支持包的片段,所以我尝试调用 getActivity().getSupportFragmentManager() 来获取 activity 支持此方法的参考。

虽然这似乎不是正确的方法。 我将这些调用更改为:

MyFragment myFragment = MyFragment.newInstance();
myFragment.setTargetFragment(ParentFragment.this, 0);
myFragment.show(getFragmentManager(), null);

因此片段自行决定使用哪个 FragmentManager,现在错误消失了。

希望这对某人有所帮助。

我遇到了与 Markus Ressel 相同的情况,但我使用的是 getChildFragmentManager()。我将其替换为 getFragmentManager() 并解决了问题。

更新:我现在一直在使用 childFragmentManager 并获得了一些反馈。

在处理由片段托管的内部片段(因此片段中的片段)时,使用 childFragmentManager。此片段管理器不同于活动 getSupportFragmentManager。两者不一样。这是关注点的分离。

所以我制定了一个规则,即托管子片段的片段将始终使用 childFragmentManager,而不在宿主片段内的东西可以使用 getSupportfragmentManager。

当我们转移到 Android Studio 3 并将支持库升级到版本 26 时,我在当前正在处理的项目中遇到了同样的问题。突然之间,没有更改代码,我们有很多这种例外。最后我发现了以下内容:

Google 今年 1 月在 v4 片段管理器的源代码中添加了一个新的 "sanity check"(不确定进入哪个版本)拒绝使用片段管理器添加片段,如果它有一个目标片段集,并且在同一个片段管理器的活动片段中找不到目标片段。补丁详细描述here

较早的版本似乎并不关心这一点。我花了几天时间来更新我们代码中的所有区域,在这些区域中,使用支持片段管理器添加的片段使用子片段管理器作为其子片段的父片段作为目标。好吧,写错代码的迟到惩罚。

我的应用程序运行良好,直到将目标版本升级到 27 然后我在调用 setTargetFragment(片段片段, int requestCode)

示例:

chooseRegionFragment.setTargetFragment(ParentFragment.this, REQUEST_CODE_CHOOSE_REGION); 

只需更改为:

chooseRegionFragment.setTargetFragment(getRootParentFragment(this), REQUEST_CODE_CHOOSE_REGION);

getRootParentFragment(this)此方法将为您找到片段的根父级

/** 
   * find root parent of fragment 
   */ 
  public static Fragment getRootParentFragment(Fragment fragment) { 
    Fragment parent = fragment.getParentFragment(); 
    if(parent == null) 
      return fragment; 
    else 
      return getRootParentFragment(parent); 
  } 

最近,我的应用程序在针对 Android O 时遇到了同样的问题。作为解决方案,请使用:

myDialogFragment.show(SettingsFragment.this.getFragmentManager(), TAG);

而不是:

myDialogFragment.show(getFragmentManager(), TAG);
// or  
myDialogFragment.show(getSupportFragmentManager(), TAG);
// or  
myDialogFragment.show(getChildFragmentManager(), TAG);

使用以下解决方案,您无需担心您正在处理哪些片段管理器,

假设您一定使用了BaseFragment。

首先创建一个接口:

public interface OnRequestUpdateListener {
void onRequestUpdate(final int requestCode, final Intent data);

void setRequestFragment(final BaseFragment requestFragment, int requestCode);

BaseFragment getRequestFragment();

int getRequestCode();

}

在您的 BaseFragment 中实现该接口

public class BaseFragment extends Fragment implements OnRequestUpdateListener {
private BaseFragment requestFragment;

private int requestCode;

@Override
public void onRequestUpdate(int requestCode, Intent data) {
 // you can implement your logic the same way you do in onActivityResult
}

@Override
public void setRequestFragment(BaseFragment requestFragment, int requestCode) {
    this.requestFragment = requestFragment;
    this.requestCode = requestCode;
}

@Override
public BaseFragment getRequestFragment() {
    return requestFragment;
}

@Override
public int getRequestCode() {
    return requestCode;
}

}

然后,将setTargetFragment 替换为setRequestFragment,并将getTargetFragment 替换为getRequestFragment。

在这里,您还可以使用 onRequestUpdate 代替 onActivityResult。

这是一个自定义解决方案,无需担心您使用的是哪个片段管理器。

使用 getFragmentManager() 而不是 getChildFragmentManager() 也可以,但它会影响 getParentFragment()。如果您不对嵌套片段使用 getChildFragmentManager(),那么您将无法通过在 child/nested 片段中使用 getParentFragment() 来获取父片段。

在开发应用程序时,我遇到了同样的问题,我通过(在 kotlin 中)解决了它

chooseRegionFragment.setTargetFragment(this@ParentFragment, REQUEST_CODE_CHOOSE_REGION)

chooseRegionFragment.show(this@ParentFragment.parentFragmentManager, TAG)

翻译成

chooseRegionFragment.setTargetFragment(ParentFragment.this, REQUEST_CODE_CHOOSE_REGION); 

chooseRegionFragment.show(ParentFragment.this.getParentFragmentManager, TAG);

在java