我们真的了解 Activity 与 Fragment 的生命周期吗?

简介:       小菜中午和朋友闲聊,被问到 Activity 与 Fragment 的生命周期,以为是很基础的东西,基本可以把生命周期说全,但是被问到 Activity 与 Fragment 交互的生命周期运行顺序,切换 Fragment 时候的生命周期顺序,以及切换完之后退出时的生命周期...瞬间感觉基础知识太不扎实了。

      小菜中午和朋友闲聊,被问到 Activity 与 Fragment 的生命周期,以为是很基础的东西,基本可以把生命周期说全,但是被问到 Activity 与 Fragment 交互的生命周期运行顺序,切换 Fragment 时候的生命周期顺序,以及切换完之后退出时的生命周期...瞬间感觉基础知识太不扎实了。
      不禁问一下小菜自己:我真的了解 Fragment 的生命周期么?


      网上对于 Activity 与 Fragment 生命周期的图表很多很详细,单独的 Activity 和 Fragment 生命周期暂不说明,下面主要研究一下两个生命周期一起时的运行周期顺序以及 Fragment 切换时的生命周期顺序


Activty 与 Fragment 生命周期执行顺序

      基本的在 Activity 中添加 Fragment 的代码就不上了,小菜觉得直接上日志来的更清楚:

  1. 初始化 Activity 与 (一个) Fragment 生命周期执行顺序:
    Activity 与 Fragment 初始化
  2. 应用切换至后台的执行顺序:
    切换后台
  3. 应用切换至前台的执行顺序:
    切回前台
  4. 销毁 Activity 与 (一个) Fragment 生命周期执行顺序:
    销毁 Fragment 与 Activity

      总结一下,其实时很容易理解的,Fragment 是依赖与 Activity 的,创建时优先执行 Activity 生命周期,销毁时优先执行 Fragment 生命周期。小菜自己画了一个小流程图,虽然很不专业,但流程基本正确。(青绿色代表 Fragment 单独生命周期,蓝色代表 Activity 单独生命周期,红色代表两者交互时生命周期)

Activity + Fragment 生命周期流程图

Fragment 切换(replace 方式)

  1. 初始化 Activity 与 FragmentAK,之后切换初始化 FragmentBK 生命周期执行顺序:


    初始化两个 Fragment
  2. 应用切换至后台的执行顺序:


    切换后台
  3. 应用切换至前台的执行顺序:


    切回前台
  4. 由 FragmentBK 切换回 FragmentAK 生命周期执行顺序:


    Fragment 之间切换
  5. 最终展示 FragmentBK 时销毁 Activity 与 (一个) Fragment 生命周期执行顺序:


    销毁 Fragment 与 Activity

      小菜个人理解,replace 方式切换 Fragment 时,每次执行 replace 方式就会销毁上一个已存在的 Fragment,即 Activity 中只包含一个 Fragment。

主要代码:(Kotlin 方式)

// 初始化加载 Fragment
fun initData() {
    if (fragmentA == null) {
        fragmentA = FragmentAK()
    }
    supportFragmentManager.beginTransaction().replace(R.id.fl_content, fragmentA).commitAllowingStateLoss()
    tv_toolbar_title.text = "FragmentAK"
    fragment = fragmentA
}
// 点击切换 Fragment
fun 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 方式)

  1. 初始化 Activity 与 FragmentAK,之后切换初始化 FragmentBK,之后再切换初始化 FragmentCK 生命周期执行顺序:


    初始化三个 Fragment
  2. 应用切换至后台的执行顺序:


    切换后台
  3. 应用切换至前台的执行顺序:


    切回前台
  4. 销毁 Activity 与 (三个) Fragment 生命周期执行顺序:


    销毁三个 Fragment 与 Activity

      如果以此方式在 Activity 中只加载两个 Fragment,简化如下:

  1. 初始化 Activity 与 (两个) Fragment,并切换至后台:


    两个 Fragment 初始化与切换
  2. 切换回前台,之后销毁 Activity 与 (两个) Fragment 生命周期执行顺序:


    两个 Fragment 销毁

      小菜个人理解,hide/show 方式切换 Fragment 时,Fragment 不销毁,其生命周期按照这几个 Fragment 的初始化顺序执行,初始化几个就执行几个 Fragment 的生命周期。

主要代码:(Kotlin 方式)

// 初始化加载 Fragment
fun 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
}
// 点击切换 Fragment
fun 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

      小菜真正认识到基础知识太重要了,得努力学扎实些啊!下面的是小菜我的公众号,欢迎闲来吐槽哦~
小菜公众号
目录
相关文章
|
8月前
activity中加载fragment的控件 在fragment 中调用activity中的控件
activity中加载fragment的控件 在fragment 中调用activity中的控件
58 0
|
8月前
fragment启动activity方法
fragment启动activity方法
60 1
|
XML 缓存 Android开发
QMUI实战(二)—Activity 和 Fragment,我们该选择谁?
在一开始,官方只提供了 Activity 来作为 UI 界面的载体,因此我们也别无选择,只能用它。而在 Android 3.0 后,Fragment 也面世了,它一开始是用于适配平板的,以邮件列表与详情的适配为例,手机端够小,因此开始展示列表,点击进入详情,而平板够大,则可以列表显示在左侧,详情显示在右侧,点击列表只是切换详情。对于这种适配场景,列表页和详情页必须在同一个 Activity 里了,而这便是我所知道的 Fragment 诞生的场景了。
214 0
|
XML ARouter API
Fragivity:像使用Activity一样使用Fragment
近年来,SPA,即单Activity架构逐渐开始受到欢迎,随之而生了很多优秀的三方库,大部分是基于Fragment作为实现方案,Fragivity 使用 Fragment + Navigatiion 打造最好用的 SPA 框架
512 0
|
API Android开发
Activity的生命周期
典型情况下的生命周期分析 Activity生命周期 activity 第一次启动 : onCreate -> onStart -> onResume 用户打开新activity,原activity的过程 : onPause -> onStop.
1443 0