美文网首页
模拟Activity意外销毁,并解决fragment重叠问题

模拟Activity意外销毁,并解决fragment重叠问题

作者: 空老表 | 来源:发表于2018-09-19 11:26 被阅读0次

前言:在APP首页很多时候都会用到多个fragment之间的切换,如果用replace来完成切换固然会避免很多不必要的问题,但是不能保存页面状态,对用户体验明显不太友好,页面重新加载也会造成卡顿,作为程序员应该尽量避免这类问题,因此本文将会详细记录这种问题的处理方案

最近测试提交了个bug,不过是很容易复现,以下是测试原话

      手机放一段时间过后滑动首页页面会出现重叠

听到重叠这两个字我心里就有数了,一般都是activity的意外销毁,导致activity保存了已添加过的fragment实例,在activity重建的时候会默认添加add到当前activity上,而我们新建的fragment根据逻辑也会add到activity之上,这就造成了fragment的重叠。
以下我首页切换fragment的代码,在add的时候添加一个当前fragment的全类名作为tag,为什么用这个当tag,后文中会提到,这里先卖个关子

    private fun switchFragment(fragment: Fragment) {
        val tag = fragment::class.java.name

        val findFragmentByTag = supportFragmentManager.findFragmentByTag(currentTag)
        if (findFragmentByTag != null && findFragmentByTag.isAdded && !findFragmentByTag.isHidden) {
            supportFragmentManager.beginTransaction().hide(findFragmentByTag).commit()
        }

        if (fragment.isAdded) {
            supportFragmentManager.beginTransaction().show(fragment).commit()
        } else {
            supportFragmentManager.beginTransaction()
                    .add(R.id.mainContainer, fragment, tag)
                    .commit()
        }
        currentTag = tag
    }

当activity意外销毁会调用onSaveInstanceState(outState: Bundle?),这里一定要看清楚,这里是一个参数,别用成了两个参数的,为什么要强调这里,当然是我之前出过错啊!!tags是fragment对应的tag集合,这里保存起来,恢复的时候就不用重新添加了

    private val _saveKey = "saveKey"

    override fun onSaveInstanceState(outState: Bundle?) {
        super.onSaveInstanceState(outState)
        outState!!.putSerializable(_saveKey,tags)
    }

在activity重建的时候还是走onCreate(savedInstanceState: Bundle?),判断里面的savedInstanceState不为null就表示是意外销毁的情况,下面的反射可以可以说是本篇的亮点了,这也是前文tag传全类名的原因所在,还是有点精髓哈,有点小开心,哈哈哈哈

    private fun initViews(savedInstanceState: Bundle?) {

        if (savedInstanceState != null) {
            val serializable = savedInstanceState.getSerializable(_saveKey) as ArrayList<String>
            tags.addAll(serializable)

            for (tag in tags) {
                var findFragmentByTag = supportFragmentManager.findFragmentByTag(tag)
                if (findFragmentByTag == null) {
                    //这里通过tag反射获取fragment的类,newInstance()获取Fragment实例
                    val loadClass = savedInstanceState.classLoader.loadClass(tag)
                    findFragmentByTag =  loadClass.newInstance() as Fragment
                }
                fragments.add(findFragmentByTag)
            }
        } else {
            tags.clear()
            tags.add(JournalFragment::class.java.name)
            tags.add(WaitingFragment::class.java.name)
            tags.add(HomeFragment::class.java.name)
            tags.add(TrucksFragment::class.java.name)
            tags.add(UserFragment::class.java.name)

            fragments.clear()
            fragments.add(JournalFragment())
            fragments.add(WaitingFragment())
            fragments.add(HomeFragment())
            fragments.add(TrucksFragment())
            fragments.add(UserFragment())
        }           
    }

至此,activity意外销毁导致的Fragment重叠问题已经完美解决,哦,还好漏了一个点,如何模拟activity销毁并支持debug,其实很简单,一张图片教会你

TIM截图20180919112243.png

只需要在开发者模式下打开不保留活动的开关,就可以了

本文写的有些粗糙,如果有不当之处,欢迎指正

相关文章

网友评论

      本文标题:模拟Activity意外销毁,并解决fragment重叠问题

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