Android中Fragment多层嵌套时onActivityResult无法正确回调问题的解决方法_Android

前言:

Fragment也可以使用startActivityForResult方法去打开一个Activity,然后在其onActivityResult方法中处理结果,可是当Fragment嵌套的时候,由于FragmentActivity的BUG导致只会回调最外那层Fragment的onActivityResult方法,于是乎当前Fragment就收不到结果了。

BUG分析:

解决这个问题之前我们先通过源码分析一下是什么原因导致的,以22.2.1版本的support-v4库为例

我们先从Fragment的startActivityForResult开始分析

public void startActivityForResult(Intent intent, int requestCode) {
  if(this.mActivity == null) {
    throw new IllegalStateException("Fragment " + this + " not attached to Activity");
  } else {
    this.mActivity.startActivityFromFragment(this, intent, requestCode);
  }
}

很明显直接调用了FragmentActivity的startActivityForFragment方法

public void startActivityFromFragment(Fragment fragment, Intent intent, int requestCode) {
  if(requestCode == -1) {
    super.startActivityForResult(intent, -1);
  } else if((requestCode & -65536) != 0) {
    throw new IllegalArgumentException("Can only use lower 16 bits for requestCode");
  } else {
    super.startActivityForResult(intent, (fragment.mIndex + 1 << 16) + (requestCode & '\uffff'));
  }
}

在这里将requestCode和Fragment的mIndex融合成了一个整型作为新的requestCode,那么新的requestCode的高16位表示Fragment的索引,低16为表示原本的requestCode,看来这里是用Fragment的mIndex作为查找依据的

接下来再看FragmentActivity的onActivityResult方法

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  this.mFragments.noteStateNotSaved();
  int index = requestCode >> 16;
  if(index != 0) {
    --index;
    if(this.mFragments.mActive != null && index >= 0 && index < this.mFragments.mActive.size()) {
      Fragment frag = (Fragment)this.mFragments.mActive.get(index);
      if(frag == null) {
        Log.w("FragmentActivity", "Activity result no fragment exists for index: 0x" + Integer.toHexString(requestCode));
      } else {
        frag.onActivityResult(requestCode & '\uffff', resultCode, data);
      }

    } else {
      Log.w("FragmentActivity", "Activity result fragment index out of range: 0x" + Integer.toHexString(requestCode));
    }
  } else {
    super.onActivityResult(requestCode, resultCode, data);
  }
}

在这里取出requestCode的高16位,不等0就是子Fragment的mIndex,但是接下来却直接从Activity的Fragment列表中根据索引去找Fragment,如果你的Fragment是被子Fragment的childFragmentManager管理的话这样是绝对找不到的,于是乎答案很明了了。

解决问题:

解决这个问题的办法有两个

第一个是直接升级support-v4到23.2.0以上的版本,因为23.2.0以上就修复了这个BUG,具体怎么解决的就不再赘述了有兴趣的可自行研究,但由于种种原因没办法升级23.2.0的还大有人在

第二种办法就是自己动手丰衣足食,在Fragment层重写相关方法解决问题

接下来着重介绍自己动手丰衣足食的办法,完整实现如下:

public class ForResultNestedCompatFragment extends Fragment {
  private ForResultNestedCompatFragment forResultChildFragment;

  @Override
  public void startActivityForResult(Intent intent, int requestCode) {
    Fragment parentFragment = getParentFragment();
    if (parentFragment != null && parentFragment instanceof ForResultNestedCompatFragment) {
      ((ForResultNestedCompatFragment) parentFragment).startActivityForResultFromChildFragment(intent, requestCode, this);
    } else {
      forResultChildFragment = null;
      super.startActivityForResult(intent, requestCode);
    }
  }

  private void startActivityForResultFromChildFragment(Intent intent, int requestCode, ForResultNestedCompatFragment childFragment) {
    forResultChildFragment = childFragment;

    Fragment parentFragment = getParentFragment();
    if (parentFragment != null && parentFragment instanceof ForResultNestedCompatFragment) {
      ((ForResultNestedCompatFragment) parentFragment).startActivityForResultFromChildFragment(intent, requestCode, this);
    } else {
      super.startActivityForResult(intent, requestCode);
    }
  }

  @Override
  public final void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (forResultChildFragment != null) {
      forResultChildFragment.onActivityResult(requestCode, resultCode, data);
      forResultChildFragment = null;
    } else {
      onActivityResultNestedCompat(requestCode, resultCode, data);
    }
  }

  public void onActivityResultNestedCompat(int requestCode, int resultCode, Intent data) {

  }
}

具体思路一句话概括就是startActivityForResult的时候一层一层晚上让父Fragment持有子Fragment的引用,回调onActivityResult的时候父Fragment再一层一层传给子Fragment。

具体用法就是让所有的Fragment都继承ForResultNestedCompatFragment,然后用onActivityResultNestedCompat方法替换onActivityResult方法。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
Fragment多层嵌套
onactivityresult回调、fragment嵌套fragment、activity嵌套fragment、fragment嵌套listview、fragment 多层嵌套,以便于您获取更多的相关知识。

时间: 2024-08-04 04:45:33

Android中Fragment多层嵌套时onActivityResult无法正确回调问题的解决方法_Android的相关文章

Android中加载网络资源时的优化可使用(线程+缓存)解决_Android

网上关于这个方面的文章也不少,基本的思路是线程+缓存来解决.下面提出一些优化: 1.采用线程池 2.内存缓存+文件缓存 3.内存缓存中网上很多是采用SoftReference来防止堆溢出,这儿严格限制只能使用最大JVM内存的1/4 4.对下载的图片进行按比例缩放,以减少内存的消耗 具体的代码里面说明.先放上内存缓存类的代码MemoryCache.java: 复制代码 代码如下: <SPAN style="FONT-SIZE: 18px"><STRONG>publ

Android中加载网络资源时的优化可使用(线程+缓存)解决

网上关于这个方面的文章也不少,基本的思路是线程+缓存来解决.下面提出一些优化: 1.采用线程池 2.内存缓存+文件缓存 3.内存缓存中网上很多是采用SoftReference来防止堆溢出,这儿严格限制只能使用最大JVM内存的1/4 4.对下载的图片进行按比例缩放,以减少内存的消耗 具体的代码里面说明.先放上内存缓存类的代码MemoryCache.java: 复制代码 代码如下: <SPAN style="FONT-SIZE: 18px"><STRONG>publ

64位系统中IIS7运行ASP时出现ADODB.Connection 800a0e7a错误的解决方法_win服务器

今天将一个ASP网站拷贝到64位的Windows7中运行,IIS7安装都没有问题,可就是死活运行不了,总是出现ADODB.Connection错误"800a0e7a",真是奇怪,之前在XP中运行一点问题都没有的.起初以为是代码上的原因,于是写一段最简单的连接数据库的代码,还是出错,研究半天终于找到了解决方法,拿来分享. 原因是因为在64位Windows7操作系统中,IIS7应用程序池默认没有启用32位应用程序,而我们连接ACCESS数据库的驱动程序Microsoft.Jet.OLEDB

Android Studio 报错failed to create jvm error code -4的解决方法_Android

安装完 Android Studio 后启动,却报错如下: 复制代码 代码如下: failed to create jvm error code -4 这一般应是内存不够用所致,解决方法参考如下. 打开 Android Studio 安装目录下的bin目录,查找并打开文件 studio.exe.vmoptions,修改代码: 复制代码 代码如下: -Xmx512m 为 -Xmx256m 保存后应即可正常打开了.

Android中在GridView网格视图上实现item拖拽交换的方法_Android

GridView基础新建一个HelloGridView的工程 修改main.xml代码如下: <?xml version="1.0" encoding="utf-8"?> <GridView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/gridview" android:layout_width=&qu

Android 软键盘弹出时把原来布局顶上去的解决方法

键盘弹出时,会将布局底部的导航条顶上去. 解决办法: 在mainfest.xml中,在和导航栏相关的activity中加: <activity android:name=".filing.MainActivity" android:windowSoftInputMode="adjustResize|stateHidden" /> windowSoftInputMode 属性解释: 活动的主窗口如何与包含屏幕上的软键盘窗口交互.这个属性的设置将会影响两件事

Android中fragment嵌套fragment问题解决方法_Android

都说fragment好用,duang~~,又遇到问题了,记录一下,分享给遇到这个问题的同学! 1.fragment嵌套fragment时出现getActivity()为null         activity A嵌套fragment B,B嵌套fragment C,C跳转到activity D,当activity D被finish掉之后,C中容易爆出getActivity为空.如果你的activity被回收了,那你需要在bundle中保存一下fragment信息,我的解决方法:fragment

Android Fragment多层嵌套重影问题的解决方法_Android

1解决bug的思想: //step1:当bug被发现(排除极低偶然性,单次性,开发工具导致) //step2:根据经验判断bug的重现场景,多次测试,直到精准的定位bug //step3:根据重现场景找到对应的代码 //step4:分析区域代码是否会影响到其他功能. //step5:做好数据的备份工作.(做好代码重构和恢复的准备,这样你才能肆无忌惮的捣鼓代码) //step6:修复代码的过程中,你会发现可能有多种解决方案.试着采取不影响主线的解决方案.以免影响到其他的代码. //step7:回顾

Android 中 Fragment 嵌套 Fragment使用存在的bug附完美解决方案_Android

自从Android3.0引入了Fragment之后,使用Activity去嵌套一些Fragment的做法也变得更加流行,这确实是Fragment带来的一些优点,比如说:Fragment可以使你能够将activity分离成多个可重用的组件,每个都有它自己的生命周期和UI,更重要的是Fragment解决了Activity间的切换不流畅,实现了一种轻量及的切换,但是在官方提供的android.support.v4包中,Fragment还是或多或少的存在一些BUG,今天就与大家分享一下这些BUG和解决方