原博客地址:https://blog.csdn.net/weixin_40400031/article/details/91127309
引言:现在一般的开发中,使用到Fragment则必不可少需要用到ViewPager。而ViewPager + Fragment 已经是Android应用中最常见的tab分页实现方式。可是ViewPager虽好,却有一个不足之处:无法自己实现懒加载!
在之前的开发中,笔者尝试过使用 :
viewpager.setOffscreenPageLimit(0);
结果:
但是并不能达到目的,这是因为ViewPager内部默认至少预加载一页,即使你设置为0,它仍然会为你预加载。
除了以上解决方案,笔者还尝试过:
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if(isVisibleToUser){
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
//do something...
}
},300);
}
}
结果:
这样做,看似可以实现懒加载的目的,但是handler的延时机制对于fragment来说非常脆弱,如果内存吃紧导致fragment绘制布局过慢,则setUserVisibleHint()方法内部对布局控件的操作就会报java.lang.NullPointerException 空指针异常,因为你的布局还没有加载完,view还没有进行实例化,故ANR崩溃闪退。
而且这样做还有一个弊端,如果你的延时时间过长,则会导致用户看到的内容卡顿;如果时间过短,很难保证在这之前所有布局都已经绘制完成。
最终解决方案
通过ViewTreeObserver.OnGlobalLayoutListener()接口获取界面绘制结束信息,配合上setUserVisibleHint(),来控制懒加载的准确和唯一。
以下是简化版的Fragment类,开发中可以在base类中封装LazyLoadFragment,提供抽象方法lazyLoad()给子类使用
/**
* 对fragment封装,实现懒加载
*/
public class LazyLoadFragment extends Fragment {
private Boolean isAlreadyInvokedLazyLoad = false; //是否已经加载过lazyLoad
View view ;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
view = LayoutInflater.from(this.getContext()).inflate(... ,container,false);
view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
if(getUserVisibleHint() && isAlreadyInvokedLazyLoad == false){
//do something ...
lazyLoad();
// 由于onGlobalLayout()会重复调用
// 所以避免多次调用,lazyLoad()一旦调用,立即将 isAlreadyInvokedLazyLoad 置为true
isAlreadyInvokedLazyLoad = true ;
}
}
});
return view;
}
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if(isVisibleToUser == false){
isAlreadyInvokedLazyLoad = false ; //隐藏之后,重新将该变量置为空
}
}
//实现懒加载
protected void lazyLoad(){
//do something ...
}
}