Fragment重影(重疊)白屏等問題原理解析,以及解決方案

前言

絕大部分的app首頁架構(gòu)均為Tab + Fragment,當(dāng)程序發(fā)生異常自動(dòng)恢復(fù),或者app長時(shí)間處于后臺(tái)恢復(fù)后,F(xiàn)ragment出現(xiàn)重影(重疊)等問題。當(dāng)然部分不顧及頁面層級(jí)的小伙伴,每個(gè)Fragment的view都設(shè)置了背景,可能就察覺不出來,但是并不代表沒有。然后很多Fragment里面又還有Fragment的使用不當(dāng)甚至?xí)霈F(xiàn)白屏的現(xiàn)象。

1 重影(重疊)

1.1 觸發(fā)原因

Activity在非正常退出(點(diǎn)返回等屬于正常退出)會(huì)調(diào)用 onSaveInstanceState 方法來保存數(shù)據(jù),其中就包括視圖層(View Hierarchy),當(dāng)該Activity在此被重建時(shí),會(huì)調(diào)用onRestoreInstanceState方法,之前被實(shí)例化過的 Fragment 依然會(huì)出現(xiàn)在 Activity 中,然后按照正常生命流程走,在onCreate中FragmentTransaction相當(dāng)于又再次 add 了 fragment 進(jìn)去的,hide()和show()方法對(duì)之前保存的fragment已經(jīng)失效了。綜上這些因素導(dǎo)致了多個(gè)Fragment重疊在一起

1.2 如何調(diào)試

  • 當(dāng)你不確定你的app是否存在該問題時(shí),先檢查fragment是否有背景,如果有,先刪掉
  • 手機(jī)的 “設(shè)置” - “開發(fā)者選項(xiàng)” - 打開”不保留活動(dòng)”(主要用于模擬Activity被及時(shí)回收)
  • 把 app 切換到后臺(tái),再重新打開,通過點(diǎn)按不同的 tab 來切換 Fragment,打開其他頁面在回來,在切換tab
  • 如果有重影,請(qǐng)接著看下面的解決方案,如果沒有,恭喜你,你的代碼太完美了,希望你能提供更優(yōu)質(zhì)的解決方案

1.3 解決方案

1.3.1 在onCreate方法判斷 savedInstanceState 參數(shù)是否為null (不推薦)

如果savedInstanceState不為null,說明該Activity有保存的實(shí)例,在add fragment 時(shí)添加標(biāo)簽,具體看源碼
selectedFragment方法 其中XXX.getClass().getSimpleName()為Tag 為演示才這樣寫的

private void selectedFragment(int position) {
        mPosition = position;
        FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
        hideFragment(transaction);
        switch (position) {
            case 0:
                if (fragment1 == null) {
                    fragment1 = new Fragment1();
                    transaction.add(R.id.fl_content, fragment1,fragment1.getClass().getSimpleName());
                } else {
                    transaction.show(fragment1);
                }
                break;
            case 1:
                if (fragment2 == null) {
                    fragment2 = new Fragment2();
                    transaction.add(R.id.fl_content, fragment2,fragment2.getClass().getSimpleName());
                } else {
                    transaction.show(fragment2);
                }
                break;
            case 2:
                if (fragment3 == null) {
                    fragment3 = new Fragment3();
                    transaction.add(R.id.fl_content, fragment3,fragment3.getClass().getSimpleName());
                } else {
                    transaction.show(fragment3);
                }
                break;
            case 3:
                if (fragment4 == null) {
                    fragment4 = new Fragment4();
                    transaction.add(R.id.fl_content, fragment4,fragment4.getClass().getSimpleName());
                } else {
                    transaction.show(fragment4);
                }
                break;
            default:
        }
        transaction.commitAllowingStateLoss();
    }

onCreate方法代碼

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.initData(savedInstanceState);
    //不為null,說明是死而復(fù)活,移除已經(jīng)存在的fragment
    if (savedInstanceState != null) {
        FragmentTransaction mTransaction = getSupportFragmentManager().beginTransaction();
        mTransaction.remove(mManager.findFragmentByTag(Fragment4.class.getSimpleName()));
        mTransaction.remove(mManager.findFragmentByTag(Fragment3.class.getSimpleName()));
        mTransaction.remove(mManager.findFragmentByTag(Fragment2.class.getSimpleName()));
        mTransaction.remove(mManager.findFragmentByTag(Fragment1.class.getSimpleName()));
        mTransaction.commitAllowingStateLoss();
    }
 
    selectedFragment(mPosition);
    ......
}
1.3.2 重寫onSaveInstanceState onRestoreInstanceState 方法 (推薦)

無需為Fragment 添加Tag 保持最開始的實(shí)現(xiàn)邏輯不動(dòng) 源碼

    **
     * 原理  去除Super 切斷原有恢復(fù)邏輯 保存位置
     * @param outState
     */
    @SuppressLint("MissingSuperCall")
    @Override
    protected void onSaveInstanceState(Bundle outState) {
        /* 記錄當(dāng)前的position */
        outState.putInt("position", mPosition);
    }

    @Override
    protected void onRestoreInstanceState(Bundle savedInstanceState) {
        super.onRestoreInstanceState(savedInstanceState);
        mPosition = savedInstanceState.getInt("position");
        selectedFragment(mPosition);
    }

2 白屏

2.1 觸發(fā)原因

當(dāng)Fragment里面嵌套Fragment時(shí),沒有使用getChildFragmentManager(),在Activity恢復(fù)后無法獲取FragmentManager內(nèi)的Fragment,從而出現(xiàn)白屏。

2.1 解決方案

Fragment嵌套Fragment時(shí),使用getChildFragmentManager()獲取事務(wù)

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

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容