小菜中午和朋友闲聊,被问到 Activity 与 Fragment 的生命周期,以为是很基础的东西,基本可以把生命周期说全,但是被问到 Activity 与 Fragment 交互的生命周期运行顺序,切换 Fragment 时候的生命周期顺序,以及切换完之后退出时的生命周期...瞬间感觉基础知识太不扎实了。
不禁问一下小菜自己:我真的了解 Fragment 的生命周期么?
网上对于 Activity 与 Fragment 生命周期的图表很多很详细,单独的 Activity 和 Fragment 生命周期暂不说明,下面主要研究一下两个生命周期一起时的运行周期顺序以及 Fragment 切换时的生命周期顺序:
Activty 与 Fragment 生命周期执行顺序
基本的在 Activity 中添加 Fragment 的代码就不上了,小菜觉得直接上日志来的更清楚:
初始化 Activity 与 (一个) Fragment 生命周期执行顺序:
Activity 与 Fragment 初始化
应用切换至后台的执行顺序:
切换后台
应用切换至前台的执行顺序:
切回前台
销毁 Activity 与 (一个) Fragment 生命周期执行顺序:
销毁 Fragment 与 Activity
总结一下,其实时很容易理解的,Fragment 是依赖与 Activity 的,创建时优先执行 Activity 生命周期,销毁时优先执行 Fragment 生命周期。小菜自己画了一个小流程图,虽然很不专业,但流程基本正确。(青绿色代表 Fragment 单独生命周期,蓝色代表 Activity 单独生命周期,红色代表两者交互时生命周期)
Activity + Fragment 生命周期流程图
Fragment 切换(replace 方式)
初始化 Activity 与 FragmentAK,之后切换初始化 FragmentBK 生命周期执行顺序:
初始化两个 Fragment
应用切换至后台的执行顺序:
切换后台
应用切换至前台的执行顺序:
切回前台
由 FragmentBK 切换回 FragmentAK 生命周期执行顺序:
Fragment 之间切换
最终展示 FragmentBK 时销毁 Activity 与 (一个) Fragment 生命周期执行顺序:
销毁 Fragment 与 Activity
小菜个人理解,replace 方式切换 Fragment 时,每次执行 replace 方式就会销毁上一个已存在的 Fragment,即 Activity 中只包含一个 Fragment。
主要代码:(Kotlin 方式)
// 初始化加载 Fragmentfun initData() { if (fragmentA == null) { fragmentA = FragmentAK() } supportFragmentManager.beginTransaction().replace(R.id.fl_content, fragmentA).commitAllowingStateLoss() tv_toolbar_title.text = "FragmentAK" fragment = fragmentA }// 点击切换 Fragmentfun clickFragmentLay(position: Int) { when (position) { 0 -> { if (fragmentA == null) { fragmentA = FragmentAK() } supportFragmentManager.beginTransaction().replace(R.id.fl_content, fragmentA).commitAllowingStateLoss() tv_toolbar_title.text = "FragmentAK" Toast.makeText(this@TestActivityK, "切换为 FragmentAK", Toast.LENGTH_SHORT).show() } 1 -> { if (fragmentB == null) { fragmentB = FragmentBK() } supportFragmentManager.beginTransaction().replace(R.id.fl_content, fragmentB).commitAllowingStateLoss() tv_toolbar_title.text = "FragmentBK" Toast.makeText(this@TestActivityK, "切换为 FragmentBK", Toast.LENGTH_SHORT).show() } } }
Fragment 切换(hide/show 方式)
初始化 Activity 与 FragmentAK,之后切换初始化 FragmentBK,之后再切换初始化 FragmentCK 生命周期执行顺序:
初始化三个 Fragment
应用切换至后台的执行顺序:
切换后台
应用切换至前台的执行顺序:
切回前台
销毁 Activity 与 (三个) Fragment 生命周期执行顺序:
销毁三个 Fragment 与 Activity
如果以此方式在 Activity 中只加载两个 Fragment,简化如下:
初始化 Activity 与 (两个) Fragment,并切换至后台:
两个 Fragment 初始化与切换
切换回前台,之后销毁 Activity 与 (两个) Fragment 生命周期执行顺序:
两个 Fragment 销毁
小菜个人理解,hide/show 方式切换 Fragment 时,Fragment 不销毁,其生命周期按照这几个 Fragment 的初始化顺序执行,初始化几个就执行几个 Fragment 的生命周期。
主要代码:(Kotlin 方式)
// 初始化加载 Fragmentfun initData() { if (fragmentA == null) { fragmentA = FragmentAK() } if (!fragmentA!!.isAdded()) { supportFragmentManager.beginTransaction().add(R.id.fl_content, fragmentA).commitAllowingStateLoss() } else { supportFragmentManager.beginTransaction().show(fragmentA).commitAllowingStateLoss() } tv_toolbar_title.text = "FragmentAK" fragment = fragmentA }// 点击切换 Fragmentfun clickFragmentLay(position: Int) { when (position) { 0 -> { if (fragmentA == null) { fragmentA = FragmentAK() } addOrShowFragment(supportFragmentManager.beginTransaction(), fragmentA!!) tv_toolbar_title.text = "FragmentAK" Toast.makeText(this@TestActivityK, "切换为 FragmentAK", Toast.LENGTH_SHORT).show() } 1 -> { if (fragmentB == null) { fragmentB = FragmentBK() } addOrShowFragment(supportFragmentManager.beginTransaction(), fragmentB!!) tv_toolbar_title.text = "FragmentBK" Toast.makeText(this@TestActivityK, "切换为 FragmentBK", Toast.LENGTH_SHORT).show() } 2 -> { if (fragmentC == null) { fragmentC = FragmentCK() } addOrShowFragment(supportFragmentManager.beginTransaction(), fragmentC!!) tv_toolbar_title.text = "FragmentCK" Toast.makeText(this@TestActivityK, "切换为 FragmentCK", Toast.LENGTH_SHORT).show() } } }// addOrShowFragment 方法fun addOrShowFragment(transaction: FragmentTransaction, frag: Fragment) { if (fragment === frag) { return } if (!frag.isAdded) { // 如果当前fragment未被添加,则添加到Fragment管理器中 transaction.hide(fragment).add(R.id.fl_content, frag).commitAllowingStateLoss() } else { transaction.hide(fragment).show(frag).commitAllowingStateLoss() } fragment = frag as Fragment }
小菜日常应用加载和切换 Fragment 的方式就这两种,应用较多的还是 hide/show 方式,因为如果在 Fragment 中执行异步线程时,若未执行完 replace 方式直接销毁会报异常,以及 getActivity() 方式会报空指针。当然如果加载很多 Fragment 时不止用到 hide/show 方式,还要涉及 Fragment 的懒加载等。
效果图 FragmentAK
效果图 FragmentBK
效果图 FragmentCK
作者:老菜和尚
共同学习,写下你的评论
评论加载中...
作者其他优质文章