老生常谈--Fragment重叠或重复创建问题分析和解决

问题描述

image.png

当Activity被系统因为内存记账销毁时再重建时,内部的fragment可能会导致显示混乱

复现

1.手机的 “设置” - “开发者选项” - 打开”不保留活动”(主要用于模拟Activity被及时回收)
2.把 app 切换到后台,再重新打开,通过点按不同的 tab 来切换 Fragment

原因

使用 Fragment 的状态保存,当系统内存不足,Fragment 的宿主 Activity 回收的时候,Fragment 的实例并没有随之被回收。Activity 被系统回收时,会主动调用 onSaveInstance() 方法来保存视图层(View Hierarchy),所以当 Activity 通过导航再次被重建时,之前被实例化过的 Fragment 依然会出现在 Activity 中,此时的 FragmentTransaction 中的相当于又再次 add 了 fragment 进去的,hide()和show()方法对之前保存的fragment已经失效了。综上这些因素导致了多个Fragment重叠在一起

解决

方式一(推荐):

Activity 中的 onSaveInstanceState() 里面有一句super.onSaveInstanceState(outState);,Google 对于这句话的解释是 “Always call the superclass so it can save the view hierarchy state”,大概意思是“总是执行这句代码来调用父类去保存视图层的状态”。通过注释掉这句话,这样主 Activity 因为种种原因被回收的时候就不会保存之前的 fragment state,也可以成功解决重叠的问题

public class MainActivity extends AppCompatActivity {
    public static final String TAG = MainActivity.class.getSimpleName();

    // 当前正在展示的Fragment
    private Fragment currentFragment;

    private Fragment mOneFragment;
    private Fragment mTwoFragment;
    private Fragment mThreeFragment;

    // 保存fragment的下标
    private int saveIndex;
    // activity的销毁标识
    private boolean isHomeActDestroy;

    private static final String EXTRA_SAVE_INDEX = "SaveIndex";
    private static final String EXTRA_IS_HOME_ACT_DESTROY = "isHomeActDestroy";


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        TextView tv_errorInfo = findViewById(R.id.tv_errorInfo);

        // 正常情况下初始化
        if (savedInstanceState == null) {
            Log.i(TAG, "执行了 ---- onCreate(Bundle savedInstanceState is 空 )");
            initFragment();
            //第一次初始化首页默认显示第一个fragment
            switchFragment(mOneFragment);
        }
        // Activity因为内存不足的时候,这里进行Fragment恢复
        else {
            Log.i(TAG, "执行了 ---- onCreate(Bundle savedInstanceState is 非空 )");
            tv_errorInfo.setText(getResources().getString(R.string.activity_main_error_info));

            restoreFragmentInstance(savedInstanceState);

            // 因为内存原因Activity退到后天被kill,再打开的时候,会重新走onCreate
            // 这时候恢复的Fragment默认恢复第一个
            // 实际中这里可以根据需求进行修改
            switchFragment(mOneFragment);
        }
    }

    private void initFragment() {
        mOneFragment = OneFragment.newInstance();
        mTwoFragment = TwoFragment.newInstance();
        mThreeFragment = ThreeFragment.newInstance();
    }

    /**
     * 采用 add show 方式切换fragment
     */
    private void switchFragment(Fragment targetFragment) {
        if (null == targetFragment) {
            Log.i(TAG, "为空");
            return;
        }
        FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
        // 先隐藏掉所有的Fragment
        hideFragment(transaction);

        // 在 show 操作
        if (!targetFragment.isAdded()) {
            transaction.add(R.id.fl_container, targetFragment, targetFragment.getClass().getSimpleName());
            transaction.show(targetFragment);
            transaction.commit();
            System.out.println("还没添加呢");
        } else {
            transaction
                    .show(targetFragment)
                    .commit();
            System.out.println("添加了( ⊙o⊙ )哇");
        }
        currentFragment = targetFragment;
    }

    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.btn_1:
                switchFragment(mOneFragment);
                break;
            case R.id.btn_2:
                switchFragment(mTwoFragment);
                break;
            case R.id.btn_3:
                switchFragment(mThreeFragment);
                break;
        }
    }

    // 隐藏 fragment
    private void hideFragment(FragmentTransaction transaction) {
        if (mOneFragment != null) {
            transaction.hide(mOneFragment);
        }
        if (mTwoFragment != null) {
            transaction.hide(mTwoFragment);
        }
        if (mThreeFragment != null) {
            transaction.hide(mThreeFragment);
        }
    }

    // 方式一:
    /*
    @Override
    protected void onSaveInstanceState(Bundle outState) {
        //super.onSaveInstanceState(outState);
    }
    */

    // 恢复fragment状态
    private void restoreFragmentInstance(Bundle savedInstanceState) {
        if (savedInstanceState != null) {

            saveIndex = savedInstanceState.getInt(EXTRA_SAVE_INDEX, 0);
            isHomeActDestroy = savedInstanceState.getBoolean(EXTRA_IS_HOME_ACT_DESTROY, false);

            FragmentManager manager = getSupportFragmentManager();
            Fragment f0 = manager.findFragmentByTag(OneFragment.class.getSimpleName());
            Fragment f1 = manager.findFragmentByTag(TwoFragment.class.getSimpleName());
            Fragment f2 = manager.findFragmentByTag(ThreeFragment.class.getSimpleName());

            // 复用
            if (f0 != null) {
                mOneFragment = f0;
            } else {
                mOneFragment = OneFragment.newInstance();
            }
            if (f1 != null) {
                mTwoFragment = f1;
            } else {
                mTwoFragment = TwoFragment.newInstance();
            }
            if (f2 != null) {
                mThreeFragment = f2;
            } else {
                mThreeFragment = ThreeFragment.newInstance();
            }
        }
    }

    // 退出的时候保存状态
    @Override
    protected void onSaveInstanceState(Bundle outState) {
        outState.putInt(EXTRA_SAVE_INDEX, saveIndex);
        outState.putBoolean(EXTRA_IS_HOME_ACT_DESTROY, true);
        super.onSaveInstanceState(outState);
    }
}

方式二:

1.给每个 Fragment 加一个 Tag
2.在 onCreate(Bundle savedInstanceState) 中判断 Bundle savedInstanceState 是否不为空
3.不为空则进行 find Tag,重新给几个 frament 赋值
这样子仍是对之前保存的 fragment 操作,成功解决了重叠的问题

@Override
protected void onSaveInstanceState(Bundle outState) {
  //如果用以下这种做法则不保存状态,再次进来的话会显示默认tab
  //总是执行这句代码来调用父类去保存视图层的状态
  //super.onSaveInstanceState(outState);
}

小结

用add和show方式切换Fragment时添加tag标识Fragment,Activity退到后天被kill掉时候,会重新走onCreate,这时判断saveInstance参数,如果不为空,则去通过标识取出事物中的fragment实例,取出的实例可能为null,则创建一个(有则复用,无则新建),这样就避免了Fragment重叠、或Fragment重复创建的问题

源码

下方留言、简信或者发邮件给我哟

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容