天天看點

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。

繼續閱讀