• 关于Fragment 不响应onActivityResult的情况分析


    大家都知道,可以通过使用 startActivityForResult() 和 onActivityResult() 方法来传递或接收参数。


    但你是否遭遇过onActivityResult()不执行或者未按预想的那样执行的情况呢?

    这里我总结了三种情况:

    1、执行startActivityForResult,没等到被调用的 Activity 返回,onActivityResult() 就被执行了。
    找了很久,终于通过小道消息得知,这与 Activity 的加载模式(launchMode)有关,该属性可以在 AndroidManifest.xml 中设置。
    原先将其设为 singleInstance,经测试,所有需要传递或接收的 Activity 不允许设置该属性,或只能设为标准模式,否则系统将在 startActivityForResult() 后直接调用 onActivityResult()。

    Note that this method should only be used with Intent protocols
    * that are defined to return a result. In other protocols (such as
    * {@link Intent#ACTION_MAIN} or {@link Intent#ACTION_VIEW}), you may
    * not get the result when you expect. For example, if the activity you
    * are launching uses the singleTask launch mode, it will not run in your
    * task and thus you will immediately receive a cancel result.

    2、两个activity传递数据和返回数据时,请求方的onActivityResult始终无响应,通过debug调试模式也没见调用该方法。查看了 各种配置和程序代码,均未发现有错误之处。后来仔细阅读API说明,恍然大悟,原来是调用startActivityForResult的参数问题,即调 用时这样:
    startActivityForResult(intent, 0);是第二个参数的问题,该参数必须大于0才能在返回值并激活onActivityResult方法

    我最开始是用的一个activity默认的常量:RESULT_OK,跟踪了代码后发现,该常量的值为-1,当然没法激活 onActivityResult方法了,随后随便修改为一个大于0的整数,程序即通跑成功。
    startActivityForResult(intent, 1); //这样就行了

    API描述:
    @requestCode If >= 0, this code will be returned in onActivityResult() when the activity exits.

     3、在TabHost的子Activity中startActivityForResult调用其他Activity时候遭遇到onActivityResult方法不响应的问题。可以通过调用Activity的getCallingActivity()查看要接受数据的Activity。

    API这么解释的:

    Return the name of the activity that invoked this activity. This is
    * who the data in {@link #setResult setResult()} will be sent to.

    举个列子,有两个ActivityA和B,A中执行startActivityForResult(1,new Intent(A,B.class));

    即由A调到B,再B执行setResult后执行getCallingActivity(),显示A。


    解决办法如下:

    1.通过父Activity启动其他Activity;

    2.实现父Activity的onActivityResult方法,将该处接收到的请求转发给当前活动的子Activity;

    3.自定义一个转发接口用来实现第2步中的转发功能;

    4.子Activity中实现第3步接口中的方法用来接收返回信息;


    代码如下:

    1.通过父Activity启动其他Activity
    1 getParent().startActivityForResult(new Intent(...), REQUEST_CODE); 
    2.实现父Activity的onActivityResult方法

    3.自定义接口
    public interface OnTabActivityResultListener {

    public void onTabActivityResult(int requestCode, int resultCode, Intent data);

    }

    4.子Activity需要实现onTabActivityResult方法
    @Override

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

    if (requestCode == REQUEST_CODE && resultCode == RESULT_OK) {

    //实现该处逻辑
            ...

    }

    }
    总结:凡是Activity本身含有上次父类的情况都不能直接使用startActivityForResult,而要通过getParent(),通过父类调用此方法。

    此种情况也适用于多重Fragment嵌套使用的情况。

    理论上使用FragmentActivity 和 Fragment组合时,在Fragment中调用startActivityForResult方法,实际上是调用Fragment所依付的Activity的startActivityForResult方法。然后Fragment
    Activity会先响应自身的onActivityResult方法,再传递到当前活动的Fragment中。此时,通过重载Fragment的onActivityResult方法可以接受返回事件。

    这里需要注意,如果重载了FragmentActivity的 onActivityResult方法,一定要调用super. onActivityResult(),否则Fragment中的 onActivityResult方法不会被调用。 

    4、FragmentActivity和Fragment组合使用,且使用多层Fragment组合(即Fragment内部有嵌套)。在底层Fragment中执行startActivityForResult方法后,Fragment的onActivityResult方法没有被调用。

    我重载了FragmentActivity的onActivityResult方法,也增加了super.onActivityResult()。通过调试发现,FragmentActivity的onActivityResult方法执行了,但是没有往下传递成功。后来翻看源码:

    FragmentActivity中的onActivityResult方法这样写的:

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    int index = requestCode>>16;
    if (index != 0) {
    index--;
    if (mFragments.mActive == null || index < 0 || index >= mFragments.mActive.size()) {
    Log.w(TAG, "Activity result fragment index out of range: 0x"
    + Integer.toHexString(requestCode));
    return;
    }
    Fragment frag = mFragments.mActive.get(index);
    if (frag == null) {
    Log.w(TAG, "Activity result no fragment exists for index: 0x"
    + Integer.toHexString(requestCode));
    }
    frag.onActivityResult(requestCode&0xffff, resultCode, data);
    return;
    }

    super.onActivityResult(requestCode, resultCode, data);
    }

  • 相关阅读:
    Moq4在.NET3.5和.NET4版本之间的差异
    TDD中的迭代
    洛谷 3413 萌数
    割点(tarjan)
    hdu-4507 吉哥系列故事——恨7不成妻
    hdu-3709 Balanced Number
    poj-3252 Round Numbers
    hdu 1007 Quoit Design 分治求最近点对
    LA 3905 Meteor 扫描线
    uva 11464
  • 原文地址:https://www.cnblogs.com/bigben0123/p/4284590.html
Copyright © 2020-2023  润新知