总结
参考:https://www.jianshu.com/p/28ca4cbe190c
参考:https://www.jianshu.com/p/11c8ced79193
1.生命周期
参考:https://blog.youkuaiyun.com/gyh790005156/article/details/79487576
2.Fragment实现原理和Back Stack
参考:https://www.jianshu.com/p/28ca4cbe190c
参考:https://www.jianshu.com/p/11c8ced79193
面试题
1. Fragment为什么被称为第五大组件
Fragment比Activity更节省内存,其切换模式也更加舒适,使用频率不低于四大组件,且有自己的生命周期,并且必须依付于Activity
2. Activity创建Fragment的方式
- 静态创建具体步骤
首先我们同样需要注册一个xml文件,然后创建与之对应的java文件,通过onCreatView()的返回方法进行关联,最后我们需要在Activity中进行配置相关参数即在Activity的xml文件中放上fragment的位置。
- 动态创建具体步骤
(1)创建待添加的碎片实例
(2)获取FragmentManager,在活动中可以直接通过调用 getSupportFragmentManager()方法得到。
(3)开启一个事务,通过调用beginTransaction()方法开启。
(4)向容器内添加或替换碎片,一般使用repalce()方法实现,需要传入容器的id和待添加的碎片实例。
(5)提交事务,调用commit()方法来完成。
3. FragmentPageAdapter和FragmentPageStateAdapter的区别
- FragmentPagerAdapter:
适用于页面较少的情况,后者适用于页面较多的情况,通过源码了解,主要查看destroyItem方法中的最后一行,mcurtransaction.remove(fragment),通过这行代码了解到,FragmentStatePagerAdapter是真正释放fragment内存,
- FragmentStatePagerAdapter
在FragmentPagerAdapter的destroyItem方法中所调用的是mcurtransaction.detach(fragment),他仅仅是将fragment的页面与activity的页面抽离开来,并没有真正的销毁fragment释放内存.
4. Fragment的生命周期
5. Fragment通信
- Fragment与Activity通信方式
1.在fragment中调用activity中的方法getActivity
2.在Activity中条用Fragment中的方法,一般常用的是接口回调,在fragment中创建接口,在activity中实现接口,这样就能完成activity中调用fragment中的方法
3.在Fragment中调用Fragment中的方法,首先先通过getactivity方法,获取activity的方法,然后通过fingFragmentById获取到另外一个fragment的方法,然后进行调用
6. ViewPager与Fragment结合使用时的懒加载问题
- 所谓的 “懒加载” 就是数据只有在Fragment对于用户可见的时才进行加载。因为ViewPager会帮我们预先初始化Fragment。由于这个特性,我们不能把数据的加载放到onCreateView方法或者onCreate方法中。
因此,我们需要判定Fragment在什么时候是处于可见的状态。一般我们通常是通过Fragment中的生命周期方法onResume来判断Fragment是否可见,但是由于ViewPager预加载的特性,Fragment即便不可见也会执行onResume方法,因此使用这个方法进行可见性的判断就行不通了。这个时候我们需要用到下面的这个方法来进行Fragment可见性的判断:
setUserVisibleHint()方法:
什么时候被调用?
- 当fragment被创建的时,setUserVisibleHint(boolean isVisibleToUser)方法会被调用,且传入参数值为false。
- 当fragment可见时,setUserVisibleHint(boolean isVisibleToUser)方法会被调用,且传入参数值为true。
- 当fragment由 可见 -> 不可见 时,setUserVisibleHint(boolean isVisibleToUser)方法会被调用,且传入参数值为false。
只需要当setUserVisibleHint(boolean isVisibleToUser)方法中的 isVisibleToUser 参数的值为true的时候我们才开始进行数据的加载
注意事项:
setUserVisibleHint(boolean isVisibleToUser)方法在Fragment的生命周期方法onCreate 之前调用的,也就是说他并不在Fragment的生命周期中。既然是在 onCreate 方法之前被调用,这样就存在许多不确定因素,如果Fragmnet的View还没有完成初始化之前,就在setUserVisibleHint()方法中进行UI的操作,这样显然会导致空指针的出现
解决方法:
我们需要对Fragment创建的View进行缓存,确保缓存的View不为空的情况下我们才可以在setUserVisibleHint方法中进行UI操作。
/**
* Fragment的基类(懒加载)
* Created by wangke on 17-8-15.
*/
public abstract class BaseLazyFragment extends RxFragment {
private String TAG;
//标记当前Fragment的可见状态
private boolean isFragmentVisible;
private boolean isFirstVisible;
//对Fragment中加载的View进行缓存
private View mRootView;
private Unbinder mUnbinder;
public BaseLazyFragment(String TAG) {
this.TAG = TAG;
}
/**
* 当Fragment添加到Activity的时候最先调用此方法
*
* @param context 上下文对象
*/
@Override
public void onAttach(Context context) {
super.onAttach(context);
//获取Fragment之间传递过来的参数
initArgs(getArguments());
}
/**
* 获取Fragment中传递过来的参数,选择重写
*
* @param arguments
*/
protected void initArgs(Bundle arguments) {
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
LogHelper.i("wwk", TAG + " ===>onCreate");
super.onCreate(savedInstanceState);
initVariable();
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
View view = inflater.inflate(getContentLayoutId(), container, false);
mUnbinder = ButterKnife.bind(this, view);
initWidget(mRootView);
initEvent();
return view;
}
@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
LogHelper.i("wwk", TAG + " ===>onViewCreated");
if (mRootView == null) {
mRootView = view;
if (getUserVisibleHint()) {
if (isFirstVisible) {
//处于可见状态并且Fragment是第一次开启
onFragmentFirstVisible();
isFirstVisible = false;
}
onFragmentVisibleChange(true);
isFragmentVisible = true;
}
}
//直接使用缓存的mRootView
super.onViewCreated(mRootView, savedInstanceState);
}
/**
* 初始化控件事件,选择重写
*/
private void initEvent() {
}
/**
* 初始化组件
*
* @param view
*/
protected abstract void initWidget(View view);
/**
* 当Fragment第一次创建的时候调用,如果不可见则isVisible的参数值为false
* 当Fragment对于用户可见时调用,此时的isVisibleToUser参数值为true
* 当Fragment当前的状态由可见变为不可见时调用,此时的isVisibleToUser参数为false
*
* @param isVisibleToUser true : 可见
* false : 不可见
*/
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
LogHelper.i("tag", TAG + " ==> setUserVisibleHint ");
super.setUserVisibleHint(isVisibleToUser);
//setUserVisibleHint()的调用在Fragment的声明周期外调用,需要保证rootView不为空的时候调用 onFragmentVisibleChange方法
if (mRootView == null) {
return;
}
//第一次被开启,并且对用户可见(这个if语句中的判断一般不会被执行到,setUserVisibleHint方法的调用时mRootView还没有被缓存)
if (isFirstVisible && isVisibleToUser) {
//当Fragment第一次可见的时候调用
onFragmentFirstVisible();
isFirstVisible = false;
}
//Fragment对于用户可见(已经不是第一次开启)
if (isVisibleToUser) {
onFragmentVisibleChange(true);
isFragmentVisible = true;
return;
}
//能执行到这里表明 Fragment对于用户已经处于不可见的状态
if (isFragmentVisible) {
//由 可见 -> 不可见
isFragmentVisible = false;
onFragmentVisibleChange(false);
}
}
/**
* 获取填充Fragment的View的id
*
* @return
*/
protected abstract int getContentLayoutId();
/**
* 当Fragment的状态发生变化的时候调用,用于进行数据的刷新
*
* @param isVisible true 不可见 -> 可见
* false 可见 -> 不可见
*/
protected abstract void onFragmentVisibleChange(boolean isVisible);
/**
* 当Fragment第一次被开启的时候调用,用于请求数据
*/
protected abstract void onFragmentFirstVisible();
/**
* 获取当前Fragment的可见状态
*
* @return
*/
protected boolean isFragmentVisible() {
return isFragmentVisible;
}
/**
* 给变量赋初始值
*/
private void initVariable() {
//标记是否是第一次开启当前Fragment
isFirstVisible = true;
//标记Fragment对于用户是否可见
isFragmentVisible = false;
//缓存Fragment创建出来的View
mRootView = null;
}
@Override
public void onDestroy() {
initVariable();
mUnbinder.unbind();
super.onDestroy();
}
}