1. 程式人生 > >Android 中 Fragment 巢狀 Fragment使用存在的bug附完美解決方案

Android 中 Fragment 巢狀 Fragment使用存在的bug附完美解決方案

原文地址:https://blog.csdn.net/u014365133/article/details/73176068

自從Android3.0引入了Fragment之後,使用Activity去巢狀一些Fragment的做法也變得更加流行,這確實是Fragment帶來的一些優點,比如說:Fragment可以使你能夠將activity分離成多個可重用的元件,每個都有它自己的生命週期和UI,更重要的是Fragment解決了Activity間的切換不流暢,實現了一種輕量及的切換,但是在官方提供的android.support.v4包中,Fragment還是或多或少的存在一些BUG,今天就與大家分享一下這些BUG和解決方法。

Case 1:當使用Fragment去巢狀另外一些子Fragment的時候,我們需要去管理子Fragment,這時候需要呼叫ChildFragmentManager去管理這些子Fragment,由此可能產生的Exception主要是: 
java.lang.IllegalStateException: No activity

首先我們來分析一下Exception出現的原因:

通過DEBUG發現,當第一次從一個Activity啟動Fragment,然後再去啟動子Fragment的時候,存在指向Activity的變數,但當退出這些Fragment之後回到Activity,然後再進入Fragment的時候,這個變數變成null,這就很容易明瞭為什麼丟擲的異常是No activity

這個Exception是由什麼原因造成的呢?如果想知道造成異常的原因,那就必須去看Fragment的相關程式碼,發現Fragment在detached之後都會被reset掉,但是它並沒有對ChildFragmentManager做reset,所以會造成ChildFragmentManager的狀態錯誤。

找到異常出現的原因後就可以很容易的去解決問題了,我們需要在Fragment被detached的時候去重置ChildFragmentManager,即:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 @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的情況。

我們來看看FragmentActivity中的程式碼:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 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存在的原因之後,就可以完美的解決問題。希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回覆大家的。在此也非常感謝大家對指令碼之家網站的支援!

        </div>
            </div>

自從Android3.0引入了Fragment之後,使用Activity去巢狀一些Fragment的做法也變得更加流行,這確實是Fragment帶來的一些優點,比如說:Fragment可以使你能夠將activity分離成多個可重用的元件,每個都有它自己的生命週期和UI,更重要的是Fragment解決了Activity間的切換不流暢,實現了一種輕量及的切換,但是在官方提供的android.support.v4包中,Fragment還是或多或少的存在一些BUG,今天就與大家分享一下這些BUG和解決方法。

Case 1:當使用Fragment去巢狀另外一些子Fragment的時候,我們需要去管理子Fragment,這時候需要呼叫ChildFragmentManager去管理這些子Fragment,由此可能產生的Exception主要是: 
java.lang.IllegalStateException: No activity

首先我們來分析一下Exception出現的原因:

通過DEBUG發現,當第一次從一個Activity啟動Fragment,然後再去啟動子Fragment的時候,存在指向Activity的變數,但當退出這些Fragment之後回到Activity,然後再進入Fragment的時候,這個變數變成null,這就很容易明瞭為什麼丟擲的異常是No activity

這個Exception是由什麼原因造成的呢?如果想知道造成異常的原因,那就必須去看Fragment的相關程式碼,發現Fragment在detached之後都會被reset掉,但是它並沒有對ChildFragmentManager做reset,所以會造成ChildFragmentManager的狀態錯誤。

找到異常出現的原因後就可以很容易的去解決問題了,我們需要在Fragment被detached的時候去重置ChildFragmentManager,即:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 @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); } }