美文网首页Android开发经验谈Android高级进阶Android技术知识
Fragment重影(重叠)白屏等问题原理解析,以及解决方案

Fragment重影(重叠)白屏等问题原理解析,以及解决方案

作者: lance_小超 | 来源:发表于2019-01-03 16:15 被阅读4次

    前言

    绝大部分的app首页架构均为Tab + Fragment,当程序发生异常自动恢复,或者app长时间处于后台恢复后,Fragment出现重影(重叠)等问题。当然部分不顾及页面层级的小伙伴,每个Fragment的view都设置了背景,可能就察觉不出来,但是并不代表没有。然后很多Fragment里面又还有Fragment的使用不当甚至会出现白屏的现象。

    1 重影(重叠)

    1.1 触发原因

    Activity在非正常退出(点返回等属于正常退出)会调用 onSaveInstanceState 方法来保存数据,其中就包括视图层(View Hierarchy),当该Activity在此被重建时,会调用onRestoreInstanceState方法,之前被实例化过的 Fragment 依然会出现在 Activity 中,然后按照正常生命流程走,在onCreate中FragmentTransaction相当于又再次 add 了 fragment 进去的,hide()和show()方法对之前保存的fragment已经失效了。综上这些因素导致了多个Fragment重叠在一起

    1.2 如何调试

    • 当你不确定你的app是否存在该问题时,先检查fragment是否有背景,如果有,先删掉
    • 手机的 “设置” - “开发者选项” - 打开”不保留活动”(主要用于模拟Activity被及时回收)
    • 把 app 切换到后台,再重新打开,通过点按不同的 tab 来切换 Fragment,打开其他页面在回来,在切换tab
    • 如果有重影,请接着看下面的解决方案,如果没有,恭喜你,你的代码太完美了,希望你能提供更优质的解决方案

    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 保持最开始的实现逻辑不动 源码

        **
         * 原理  去除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 触发原因

    当Fragment里面嵌套Fragment时,没有使用getChildFragmentManager(),在Activity恢复后无法获取FragmentManager内的Fragment,从而出现白屏。

    2.1 解决方案

    Fragment嵌套Fragment时,使用getChildFragmentManager()获取事务

    相关文章

      网友评论

        本文标题:Fragment重影(重叠)白屏等问题原理解析,以及解决方案

        本文链接:https://www.haomeiwen.com/subject/uxxqrqtx.html