FragmentActivity重啟時Fragment狀態(tài)異常的問題解決辦法

在很舊的v4包里,fragment的問題更多,比如需要自己重寫Fragment類的onDestory,不過這些都在最新的v4包里解決了,但是fragment仍然問題很多,有個很常見的問題就是Activity因內存不夠被回收以后,在FragmentActivity重啟時,F(xiàn)ragment會出一些問題,比如fragment重疊,或者某些fragment不顯示之類的,這種情況原因如下:
先貼兩段代碼:

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(getContentViewResId());
 
        if (getIntent().hasExtra(Intent.EXTRA_TITLE)) {
            setTitle(getIntent().getStringExtra(Intent.EXTRA_TITLE));
        }
 
        final String customTitle = getIntent().getStringExtra(Intent.EXTRA_TITLE);
        setTitle(customTitle != null ? customTitle : getTitle());
 
        if (savedInstanceState == null) {
            mFragment = onCreatePane();
            mFragment.setArguments(intentToFragmentArguments(getIntent()));
            getFragmentManager().beginTransaction()
                    .add(R.id.root_container, mFragment, "single_pane")
                    .commit();
        } else {
            mFragment = getFragmentManager().findFragmentByTag("single_pane");
        }
    }

     * 
     * Save all appropriate fragment state.
     */
    @Override
    protected void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        Parcelable p = mFragments.saveAllState();
        if (p != null) {
            outState.putParcelable(FRAGMENTS_TAG, p);//這里保存了Fragment數(shù)據
        }
        if (mPendingFragmentActivityResults.size() > 0) {
            outState.putInt(NEXT_CANDIDATE_REQUEST_INDEX_TAG, mNextCandidateRequestIndex);
 
            int[] requestCodes = new int[mPendingFragmentActivityResults.size()];
            String[] fragmentWhos = new String[mPendingFragmentActivityResults.size()];
            for (int i = 0; i < mPendingFragmentActivityResults.size(); i++) {
                requestCodes[i] = mPendingFragmentActivityResults.keyAt(i);
                fragmentWhos[i] = mPendingFragmentActivityResults.valueAt(i);
            }
            outState.putIntArray(ALLOCATED_REQUEST_INDICIES_TAG, requestCodes);
            outState.putStringArray(REQUEST_FRAGMENT_WHO_TAG, fragmentWhos);
        }

從源碼中得知,當Activity調用onSaveInstanceState方法的時候,會保存當前Activity里面的所有Fragment,保存在了一個Bundle里,key就是FRAGMENTS_TAG。

這個Bundle會在Activity恢復的時候傳給onCreate,這就是為什么onSaveInstanceState不是null了。

如果要原樣恢復Fragment需要做的工作挺多的,需要在FragmentActivity的onCreate方法里判斷如果savedInstanceState不為空就從FragmentManager里獲取之前的實例,不過不需要再次添加到FragmentManager里了,獲取的目的是以后做什么操作還用這個獲取到的實例,雖然恢復的Fragment中的View沒問題,不過里面如果有自己的一些值并且沒有在Fragment自己的savedInstanceState方法里保存的話,恢復的fragment是沒有這些值的(如果這些值是從其他頁面通過bundle傳遞過來的那么你獲取的時候肯定是從bundle獲取這時候是有值的),我覺得在大部分應用里不需要這么做,有兩種更簡單的解決辦法:

  1. 重寫FragmentActivity的onCreate方法:
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        //刪除保存的Fragment狀態(tài)
        if (savedInstanceState != null) {
           //"android:support:fragments"是FragmentActivity類里的包可見的值,因為不是public的所以直接寫實際的字符串
          savedInstanceState.putParcelable("android:support:fragments", null);
        }
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mFragment = new onCreatePane();
        getSupportFragmentManager().beginTransaction()
                .add(R.id. root_container, mFragment, "single_pane")
                .commit();
    }
  1. 重寫FragmentActivity的onSaveInstanceState方法:

    override fun onSaveInstanceState(outState: Bundle) {
        super.onSaveInstanceState(outState)
        //刪除保存的Fragment狀態(tài)
        outState.putParcelable("android:support:fragments", null)
    }

這兩種方法任選一種都可以解決該類型的問題,第1種方法是恢復的時候清空之前的fragment。第2種方法是保存的時候直接不保存。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容