Fragment Or DialogFragment Can not perform this action after onSaveInstanceState

轉載自Fragment Or DialogFragment Can not perform this action after onSaveInstanceState

表現

可會造成app崩潰掉,具體日誌如下:

異常如下:

java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
            at android.app.FragmentManagerImpl.checkStateLoss(FragmentManager.java:1323)
            at android.app.FragmentManagerImpl.enqueueAction(FragmentManager.java:1341)
            at android.app.BackStackRecord.commitInternal(BackStackRecord.java:597)
            at android.app.BackStackRecord.commit(BackStackRecord.java:575)
            at android.app.DialogFragment.show(DialogFragment.java:230)
            at com.github.afeita.net.ext.TipsingNetCallback.onStart(TipsingNetCallback.java:55)
            at com.github.afeita.net.ext.AfeitaNet$3.onStart(AfeitaNet.java:567)
            at com.github.afeita.net.ext.request.CacheRequest.deliverOnStart(CacheRequest.java:260)
            at com.github.afeita.net.ExecutorDelivery$4.run(ExecutorDelivery.java:116)
            at android.os.Handler.handleCallback(Handler.java:733)
            at android.os.Handler.dispatchMessage(Handler.java:95)
            at android.os.Looper.loop(Looper.java:136)
            at android.app.ActivityThread.main(ActivityThread.java:5001)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:515)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:785)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:601)
            at dalvik.system.NativeStart.main(Native Method)

場景

這與不洽當的使用Fragment有關係,其實不光時DialogFragment,Fragment也有的。當:

  • A activity頁面中,需要異步通知B activity去更新或do something然後A再do 自己的(A)的something,而這讓B do something洽好是切換Fragment的話(此時又回到A中自己在do something),那麼就可以出現這個異常。
  • activity頁面 某個原因要切換到後臺中了,系統調用了onSaveInstanceState,此時之後異步任務來了,是需要show或dismiss一個DialogFragment也會報這個異常。

總之是:在onSaveInstanceState後執行了commit拋出的。

原因

上面的異常,跟蹤上去是DialogFragment.show方法中報出來,好麼查看下源碼,這個show裏用做什麼,拋出這個異常的呢。

public void show(FragmentManager manager, String tag) {
        mDismissed = false;
        mShownByMe = true;
        FragmentTransaction ft = manager.beginTransaction();
        ft.add(this, tag);
        ft.commit(); //注意這裏使用提commit
    }

解決方法:

對於肯定會出現這種需求的,比如網絡請求訪問彈出的加載中…提示DialogFragment。
那麼,不採用系統默認的show方法 顯示對話框。採用自己用 FragmentTransaction控制,示例如下:

FragmentTransaction ft = activity.getFragmentManager().beginTransaction();
                ft.add(dialogFragment, this.getClass().getSimpleName());
                ft.commitAllowingStateLoss();//注意這裏使用commitAllowingStateLoss()

根據google對ft.commitAllowingStateLoss方法的說明:Like commit() but allows the commit to be executed after an activity’s state is saved. This is dangerous because the commit can be lost if the activity needs to later be restored from its state, so this should only be used for cases where it is okay for the UI state to change unexpectedly on the user.
這個方法是允許activity在state狀態改變保存(onSaveInstanceState) 時允許commit。。。但也許不是很洽當,因爲當activity onRestoreInstanceState恢復狀態時commit可能會被丟掉了。恢復不了那次的commit了。使用這個方法應該確保存頁面狀態的改變對用戶無感時。一般可以確認是即使頁面需要重新onRestoreInstanceState時上次commit丟失的也不用管時就可以用了。

注意前面雖然說的是DialogFragment,也是Fragment因爲這點並不是DialogFrament特例它屬於Fragment不能在onSaveInstanceState,普通的commit。

若DialogFragment使用了ft.commitAllowingStateLoss,那麼在關閉時使用dialogFragment.dismissAllowingStateLoss。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章