淺談Android Fragment嵌套使用存在的一些BUG以及解決方法

http://www.tuicool.com/articles/2eM32a html

原文  http://my.eoe.cn/916054/archive/24053.htmljava

自從Android3.0引入了Fragment以後,使用Activity去嵌套一些Fragment的作法也變得更加流行,這確實是Fragment帶來的一些優勢,好比說:Fragment可使你可以將activity分離成多個可重用的組件,每一個都有它本身的生命週期和UI,更重要的是Fragment解決了Activity間的切換不流暢,實現了一種輕量及的切換,可是在官方提供的android.support.v4包中,Fragment仍是或多或少的存在一些BUG,今天就與你們分享一下這些BUG和解決方法。android

Case 1:當使用Fragment去嵌套另一些子Fragment的時候,咱們須要去管理子Fragment,這時候須要調用ChildFragmentManager去管理這些子Fragment,由此可能產生的Exception主要是: 
java.lang.IllegalStateException: No activityide

首先咱們來分析一下Exception出現的緣由: 
經過DEBUG發現,當第一次從一個Activity啓動Fragment,而後再去啓動子Fragment的時候,存在指向Activity的變量,但當退出這些Fragment以後回到Activity,而後再進入Fragment的時候,這個變量變成null,這就很容易明瞭爲何拋出的異常是No activityui

這個Exception是由什麼緣由形成的呢?若是想知道形成異常的緣由,那就必須去看Fragment的相關代碼,發現Fragment在detached以後都會被reset掉,可是它並無對ChildFragmentManager作reset,因此會形成ChildFragmentManager的狀態錯誤。this

找到異常出現的緣由後就能夠很容易的去解決問題了,咱們須要在Fragment被detached的時候去重置ChildFragmentManager,即:spa

@Override  public void onDetach() {    super.onDetach();    try {      Field childFragmentManager = Fragment.class          .getDeclaredField("mChildFragmentManager");      childFragmentManager.setAccessible(true);      childFragmentManager.set(this, null);    } catch (NoSuchFieldException e) {      throw new RuntimeException(e);    } catch (IllegalAccessException e) {      throw new RuntimeException(e);    }  }

Case 2:當咱們從一個Activity啓動了一個Fragment,而後在這個Fragment中又去實例化了一些子Fragment,在子Fragment中去有返回的啓動了另一個Activity,即經過startActivityForResult方式去啓動,這時候形成的現象會是,子Fragment接收不到OnActivityResult,若是在子Fragment中是以getActivity.startActivityForResult方式啓動,那麼只有Activity會接收到OnActivityResult,若是是以getParentFragment.startActivityForResult方式啓動,那麼只有父Fragment能接收(此時Activity也能接收),但不管如何子Fragment接收不到OnActivityResult。設計

這是一個很是奇怪的現象,按理說,應該是讓子Fragment接收到OnActivityResult纔對,到底是什麼形成的呢?這是因爲某位寫代碼的員工抱怨沒發獎金,稍稍偷懶了,少寫了一部分代碼,沒有考慮到Fragment再去嵌套Fragment的狀況。code

咱們來看看FragmentActivity中的代碼:htm

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())) {
        Log.w("FragmentActivity", "Activity result fragment index out of range: 0x" + Integer.toHexString(requestCode));

        return;
      }
      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 & 0xFFFF, resultCode, data);
      }
      return;
    }

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

很顯然,設計者把Fragment的下標+1左移16位來標記這個request是否是Fragment的,拿到result再解碼出下標,直接取對應的Fragment,這樣並無去考慮對Fragment嵌套Fragment作一個Map映射,因此出現了這種BUG。

可是若是咱們須要在OnActivityResult的時候處理一些事情的話,咱們能夠經過在子Fragment中以getParentFragment.startActivityForResult的方式來啓動,而後在父Fragment中去接收數據,咱們須要在子Fragment中提供一個方法,如:getResultData(Object obj),經過父Fragment中的子Fragment實例去調用這個方法,把相應的數據傳過去,而後去更新子Fragment。

以上是在使用Fragment去嵌套Fragment的時候可能會遇到的BUG,瞭解了BUG存在的緣由以後,就能夠完美的解決問題。

相關文章
相關標籤/搜索