1. Fragment的生命周期
2. 添加Fragment的两种方式
- add 和 hide , show 配合使用
getFragmentManager().beginTransaction().add().commit()
FrameLayout container = (FrameLayout)findViewById(R.id.container);
Fragment1 fragment1 = new Fragment1();
Fragment2 fragment2 = new Fragment2();
Fragment3 fragment3 = new Fragment3();
getSupportFragmentManager().beginTransaction()
.add(R.id.container, fragment1,"fragment1") //往Activity中添加一个Fragment
.add(R.id.container,fragment2,"fragment2")
.add(R.id.container,fragment3,"fragment3")
// .hide(fragment1) //隐藏当前的Fragment,仅仅是设为不可见,并不会销毁
// .show(fragment1) //显示之前隐藏的Fragment
.commit();
- replace
getSupportFragmentManager().beginTransaction()
.replace(R.id.container,fragment3)
.commit();
- remove 从Activity中移除一个Fragment,如果被移除的Fragment没有添加到回退栈,这个Fragment实例将会被销毁
getSupportFragmentManager().beginTransaction()
.remove(fragment3)
.commit();
3. Fragment怎么携带数据?
- setArguments 传递数据
Fragment1 fragment1 = new Fragment1();
Bundle bundle = new Bundle();
bundle.putString("key","value");
fragment1.setArguments(bundle);
- getArguments 获取数据 (在 Fragment 中的 onCreate() 方法中)
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
Bundle bundle = getArguments();
bundle.getString("key");
super.onCreate(savedInstanceState);
}
4. 宿主Activity对象的优化
- 如果你需要在Fragment中用到宿主Activity对象,建议在你的基类Fragment定义一个Activity的全局变量,在onAttach中初始化,这不是最好的解决办法,但这可以有效避免一些意外Crash
protected Activity mActivity;
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
this.mActivity = activity;
}
5. getActivity()空指针
可能你遇到过getActivity()返回null,或者平时运行完好的代码,在“内存重启”之后,调用getActivity()的地方却返回null,报了空指针异常。
大多数情况下的原因:你在调用了getActivity()时,当前的Fragment已经onDetach()了宿主Activity。
比如:你在pop了Fragment之后,该Fragment的异步任务仍然在执行,并且在执行完成后调用了getActivity()方法,这样就会空指针。
解决办法:
更”安全”的方法:(对于Fragment已经onDetach这种情况,我们应该避免在这之后再去调用宿主Activity对象,比如取消这些异步任务,但我们的团队可能会有粗心大意的情况,所以下面给出的这个方案会保证安全)
在Fragment基类里设置一个Activity mActivity的全局变量,在onAttach(Activity activity)里赋值,使用mActivity代替getActivity(),保证Fragment即使在onDetach后,仍持有Activity的引用(有引起内存泄露的风险,但是相比空指针闪退,这种做法“安全”些),即:
protected Activity mActivity;
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
this.mActivity = activity;
}
/**
* 如果你用了support 23的库,上面的方法会提示过时,有强迫症的小伙伴,可以用下面的方法代替
*/
@Override
public void onAttach(Context context) {
super.onAttach(context);
this.mActivity = (Activity)context;
}
6. Fragment 与 Fragment 之间的通信
- 双重回调 (一个Fragment设置接口回调给寄主 , 然后寄主传递数据给另一个Fragment)
interface onFragmentChangeListener {
abstract void onFragmentChange(XXX xxx);
}
public void setOnFragmentChangeListener(onFragmentChangeListener l) {
this.mOnFragmentChangeListener = l;
}
EventBus (可参考 https://github.com/greenrobot/EventBus)
- 随便定义一个类
public class MessageEvent {} - 注册subscribers,接收消息
eventBus.register(this); - 定义接收的方法
@Subscribe
public void onEvent(AnyEventType event) {}; - 发送消息
eventBus.post(event);
- 随便定义一个类
7. FragmentPagerAdapter 和 FragmentStatePagerAdapter
FragmentPagerAdapter (自带缓存功能)
对于不再需要的fragment,选择调用detach方法,仅销毁视图,并不会销毁fragment实例。FragmentStatePagerAdapter
会销毁不再需要的fragment,当当前事务提交以后,会彻底的将fragmeng从当前Activity的FragmentManager中移除,state标明,销毁时,会将其onSaveInstanceState(Bundle outState)中的bundle信息保存下来,当用户切换回来,可以通过该bundle恢复生成新的fragment,也就是说,你可以在onSaveInstanceState(Bundle outState)方法中保存一些数据,在onCreate中进行恢复创建。如上所说,使用FragmentStatePagerAdapter当然更省内存,但是销毁新建也是需要时间的。一般情况下,如果你是制作主页面,就3、4个Tab,那么可以选择使用FragmentPagerAdapter,如果你是用于ViewPager展示数量特别多的条目时,那么建议使用FragmentStatePagerAdapter。
8. 管理 Fragment 的回退栈
将一个fragment添加到回退栈
getSupportFragmentManager().beginTransaction().addToBackStack(String name);获取回退栈中实体数量
getSupportFragmentManager().getBackStackEntryCount();根据name立刻弹出栈顶的fragment
getSupportFragmentManager().popBackStack(String name, int flags);根据id立刻弹出栈顶的fragment
getSupportFragmentManager().popBackStack(int id, int flags);让栈顶的fragment出栈
getSupportFragmentManager().popBackStack();管理回退栈的代码实现
只有在程序运行时被动态添加的fragment才会被添加到后退栈
getSupportFragmentManager().beginTransaction()
.add(R.id.container, fragment2)
.addToBackStack(null)
.commit();
按返回键的时候就依次弹出栈getSupportFragmentManager().popBackStack();
可通过 getFragmentManager().getBackStackEntryCount();
得到回退栈中的当前总的个数,每添加一次回退栈该数会加1.
最后 , 可以参考 onBackPressed() 的父类的源码 , 实现该功能.
9. 关于 findFragmentById() 和 findFragmentByTag()
findFragmentById(int id)
通过Fragment的ID(也可以是容器的)找到Fragment,这个ID可以是XML中的也可以是通过事务动态添加进去的。findFragmentByTag(String tag)
通过Fragment的Tag找到Fragment,这个Tag可以是XML中的也可以是通过事务动态添加进去的。如果获取的值一直为 null , 那么很可能是碎片还没加载就调用了
10. 有时应用程序退出到后台,返回回来出现了一些空白的页面?
- 很有可能是当 activity 退出后 , 由于系统资源的不足或其他原因 , 会对 fragment 进行了回收处理 , 导致空白