前言
絕大部分的app首頁架構均為Tab + Fragment,當程序發(fā)生異常自動恢復孤澎,或者app長時間處于后臺恢復后届氢,F(xiàn)ragment出現(xiàn)重影(重疊)等問題。當然部分不顧及頁面層級的小伙伴覆旭,每個Fragment的view都設置了背景退子,可能就察覺不出來,但是并不代表沒有型将。然后很多Fragment里面又還有Fragment的使用不當甚至會出現(xiàn)白屏的現(xiàn)象寂祥。
1 重影(重疊)
1.1 觸發(fā)原因
Activity在非正常退出(點返回等屬于正常退出)會調用 onSaveInstanceState 方法來保存數據,其中就包括視圖層(View Hierarchy),當該Activity在此被重建時茶敏,會調用onRestoreInstanceState方法壤靶,之前被實例化過的 Fragment 依然會出現(xiàn)在 Activity 中,然后按照正常生命流程走惊搏,在onCreate中FragmentTransaction相當于又再次 add 了 fragment 進去的贮乳,hide()和show()方法對之前保存的fragment已經失效了。綜上這些因素導致了多個Fragment重疊在一起
1.2 如何調試
- 當你不確定你的app是否存在該問題時恬惯,先檢查fragment是否有背景向拆,如果有,先刪掉
- 手機的 “設置” - “開發(fā)者選項” - 打開”不保留活動”(主要用于模擬Activity被及時回收)
- 把 app 切換到后臺酪耳,再重新打開浓恳,通過點按不同的 tab 來切換 Fragment,打開其他頁面在回來碗暗,在切換tab
- 如果有重影颈将,請接著看下面的解決方案,如果沒有言疗,恭喜你晴圾,你的代碼太完美了,希望你能提供更優(yōu)質的解決方案
1.3 解決方案
1.3.1 在onCreate方法判斷 savedInstanceState 參數是否為null (不推薦)
如果savedInstanceState不為null噪奄,說明該Activity有保存的實例死姚,在add fragment 時添加標簽,具體看源碼
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,說明是死而復活勤篮,移除已經存在的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 保持最開始的實現(xiàn)邏輯不動 源碼
**
* 原理 去除Super 切斷原有恢復邏輯 保存位置
* @param outState
*/
@SuppressLint("MissingSuperCall")
@Override
protected void onSaveInstanceState(Bundle outState) {
/* 記錄當前的position */
outState.putInt("position", mPosition);
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
mPosition = savedInstanceState.getInt("position");
selectedFragment(mPosition);
}
2 白屏
2.1 觸發(fā)原因
當Fragment里面嵌套Fragment時都毒,沒有使用getChildFragmentManager(),在Activity恢復后無法獲取FragmentManager內的Fragment,從而出現(xiàn)白屏碰缔。
2.1 解決方案
Fragment嵌套Fragment時,使用getChildFragmentManager()獲取事務