有 android 开发经验的朋友们,对 Fragment 可能都有一种又爱又恨的感觉。所以在讲 Fragment 之前我先分享一篇 Square 的文章 从今天开始抛弃 Fragment 吧! 大家看完这篇文章后再考虑要不要继续读下去。
什么是 Fragment
Fragment 是 Android 3.0 引入的 API,号称是为了解决屏幕碎片化和帮助重用代码的构造,相比 Activity 更轻量级、更灵活,使用姿势一般有以下 2 种:
1. 静态写在 xml 中
2. 代码里动态生成
Fragment 生命周期
Fragment 的生命周期如图所示:onAttach() -> onCreate() -> onCreateView() -> onActivityCreated() -> onStart() -> onResume() -> onPause() -> onStop() -> onDestroyView() -> onDestroy() -> onDetach() 。
我们看出来 Fragment 的生命周期比 Activity 要复杂很多,为了更好的理解 Fragment 的生命周期,我们做一个小测试:启动 Activity1 和 Fragment1;在 Fragment1 点击 Button 启动 Activity2 和 Fragment2;点击 back 键退出 Activity2 和 Fragment2;按 home 键退出。我们看一下分别执行了 Activity 和 Fragment 哪些生命周期函数。
1. 启动 Activity1 和 Fragment1
Activity1.onCreate() -> Fragment1.onAttach() -> Fragment1.onCreate() -> Fragment1.onCreateView() -> Fragment1.onActivityCreated() -> Fragment1.onStart() -> Activity1.onStart() -> Fragment1.onResume() -> Activity1.onResume()
2. 在 Fragment1 点击 Button 启动 Activity2 和 Fragment2
Fragment1.onPause() -> Activity1.onPause() -> Activity2.onCreate() -> Fragment2.onAttach() -> Fragment2.onCreate() -> Fragment2.onCreateView() -> Fragment2.onActivityCreated() -> Fragment2.onStart() -> Activity2.onStart() -> Fragment2.onResume() -> Activity2.onResume() -> Fragment1.onStop() -> Activity1.onStop()
3. 点击 back 键退出 Activity2 和 Fragment2
Fragment2.onPause() -> Activity2.onPause() -> Activity1.onRestart() ->Fragment1.onStart() -> Activity1.onStart() -> Activity1.onResume() -> Fragment1.onResume() -> Fragment2.onStop() -> Activity2.onStop() -> Fragment2.onDestroyView() -> Fragment2.onDestroy() -> Fragment2.onDetach() -> Activity2.onDestroy()
4. 按 home 键退出
Fragment1.onPause() -> Activity1.onPause() -> Fragment1.onStop() -> Activity1.onStop()
FragmentManager
FragmentManager 是为了管理 Activity 中的 Fragments 。
getChildFragmentManager()、getFragmentManager()、getSupportFragmentManager() 的区别:
1. getFragmentManager() 和 getSupportFragmentManager() 的区别很容易理解,android 的 v4 扩展包中的 FragmentActivity 中获取 FragmentManager 使用的就是 getSupportFragmentManager(),android.app 中获取管理类的方法就是getFragmentManager()。
2. getFragmentManager() 所得到的是所在 Fragment 的父容器的管理器,getChildFragmentManager() 所得到的是在 Fragment 里面子容器的管理器。Fragment 嵌套Fragment 要用 getChildFragmentManager。
FragmentTransaction
FragmentTransaction 封装了一系列对 Fragment 的操作,如:add/remove/replace/show/hide并一次性执行这些操作。
commit() 和 commitAllowingStateLose() 区别
大家在是用 Fragment 的时候可能都遇到过如下异常:
java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
我们可以直接看这篇 文章 。commit() 和 commitAllowingStateLoss() 的实现几乎是一样的。唯一的区别就是在调用 commit() 的时候 FragmentManager 会检查是否已经保存了其状态,如果状态已经保存了,则就抛出 IllegalStateException 异常。
commit()、commitNow() 和 executePendingTransaction() 区别
1. commit() 不会立马执行,而是等到下次主线程准备好时执行。
2. commitNow() 只同步的执行当前的提交操作。
3. executePendingTransactions() 则会执行所有等得执行的操作。
Fragment 有一个常见的问题,即 Fragment 重叠问题,这是由于 Fragment 被系统杀掉,并重新初始化时再次将 fragment 加入 activity,因此通过在外围加if语句能判断此时是否是被系统杀掉并重新初始化的情况。