存在的问题

1. 默认情况下,ViewPager会根据setOffscreenPageLimit()方法设置的大小,自动预加载
2. 还是根据setOffscreenPageLimit()方法设置的大小,会去销毁fragment视图

下面的图说明情况

滑动fragment1,此时会预加载fragment2,滑动到fragment2会预加载fragment3,但是滑动到fragment3,此时会调用fragment1的destroyview方法,销毁视图。当重新滑动到fragment1才会重新调用fragment1的oncreateview方法。注意此时并不会销毁实例,不会调用ondestroy方法

这样就存在两个问题
1.pagerview频繁切换,导致fragment1.fragment3在频繁的调用destroyview和oncreateview方法,重新创建视图。这样也浪费了大量的资源,用户体验不佳,虽然内存消耗比较低
2.因为切换到fragment1的时候,同时预加载了fragment2,如果此时fragment2也有大量的耗时网络请求要做,如果应用对启动反应速度比较敏感,所以此时做了多余的工作。能否把这些耗时的工作延迟加载,也是个问题

解决方案

1.  防止频繁的销毁视图,setOffscreenPageLimit(2)/或者重写PagerAdaper的destroyItem方法为空即可

setOffscreenPageLimit(2)

//Set the number of pages that should be retained to either side of the current page in the view hierarchy in an idle state.Pages beyond this limit will be recreated from the adapter when needed.You should keep this limit low, especially if your pages have complex layouts. This setting defaults to 1.

大概意思就是说:

设置当前page左右两侧应该被保持的page数量,超过这个限制,page会被销毁重建(只是销毁视图),onDestroy-onCreateView,但不会执行onDestroy。尽量维持这个值小,特别是有复杂布局的时候,因为如果这个值很大,就会占用很多内存,如果只有3-4page的话,可以全部保持active,可以保持page切换的顺滑

这下很好理解了,默认情况下是1,所以当前fragment左右两侧,就会被保持1页pager,所以上述切换到fragment2并不会销毁任何视图,但是到fragment1,3会。这里注意这个值,是左右两侧能够维持的page,所以如果setOffscreenPageLimit(2),那么就不会频繁的销毁了

destroyItem()

//super.destroyItem(Container, position, object);  注释掉调用父类方法即可

2.  取消预加载,可以fragment的setUserVisibleHint实现,具体实现参考代码示例

setUserVisibleHint

//Set a hint to the system about whether this fragment's UI is currently visible to the user. This hint defaults to true and is persistent across fragment instance state save and restore.An app may set this to false to indicate that the fragment's UI is scrolled out of visibility or is otherwise not directly visible to the user. This may be used by the system to prioritize operations such as fragment lifecycle updates or loader ordering behavior.

大概意思就是:fragment对用户可见,isVisibleToUser为true,不可见isVisibleToUser为false。对应于viewpager,当前pager,非当前pager

代码:

public abstract class BaseFragment extends Fragment {  
    /** Fragment当前状态是否可见 */  
    protected boolean isVisible;  
  
    //setUserVisibleHint  adapter中的每个fragment切换的时候都会被调用,如果是切换到当前页,那么isVisibleToUser==true,否则为false  
    @Override  
    public void setUserVisibleHint(boolean isVisibleToUser) {  
        super.setUserVisibleHint(isVisibleToUser);  
        if(isVisibleToUser) {  
            isVisible = true;  
            onVisible();  
        } else {  
            isVisible = false;  
            onInvisible();  
        }  
    }  
      
    /** 
     * 可见 
     */  
    protected void onVisible() {  
        lazyLoad();       
    }  
          
    /** 
     * 不可见 
     */  
    protected void onInvisible() {        
    }  
      
    /** 
     * 延迟加载 
     * 子类必须重写此方法 
     */  
    protected abstract void lazyLoad();  
}

  

public class CustomListFragment extends BaseFragment {  
  
    private Context context;  
  
    private static final String FRAGMENT_INDEX = "fragment_index";  
    private final int FIRST_FRAGMENT = 0;  
    private final int SECOND_FRAGMENT = 1;  
    private final int THIRD_FRAGMENT = 2;  
  
    private TextView contentText;  
    private ProgressBar progressBar;  
  
    private int mCurIndex = -1;  
    /** 
     * 标志位,标志已经初始化完成 
     */  
    private boolean isPrepared;  
    /** 
     * 是否已被加载过一次,第二次就不再去请求数据了 
     */  
    private boolean mHasLoadedOnce;  
  
    /** 
     * 创建新实例 
     * 
     * @param index 
     * @return 
     */  
    public static CustomListFragment newInstance(int index) {  
        Bundle bundle = new Bundle();  
        bundle.putInt(FRAGMENT_INDEX, index);  
        CustomListFragment fragment = new CustomListFragment();  
        fragment.setArguments(bundle);  
        return fragment;  
    }  
  
    @Override  
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        context = getActivity();  
  
        Bundle bundle = getArguments();  
        if (bundle != null) {  
            mCurIndex = bundle.getInt(FRAGMENT_INDEX);  
        }  
    }  
  
    @Override  
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {  
        View view = inflater.inflate(R.layout.fragment, container, false);  
        contentText = (TextView) view.findViewById(R.id.content);  
        progressBar = (ProgressBar) view.findViewById(R.id.progressbar);  
        isPrepared = true;  
        lazyLoad();  
        return view;  
    }  
  
    @Override  
    protected void lazyLoad() {  
        if (!isPrepared || !isVisible || mHasLoadedOnce) {  
            return;  
        }  
        new AsyncTask<Void, Void, Boolean>() {  
            @Override  
            protected void onPreExecute() {  
                super.onPreExecute();  
                progressBar.setVisibility(View.VISIBLE);  
            }  
            @Override  
            protected Boolean doInBackground(Void... params) {  
                try {  
                    Thread.sleep(2000);  
                    //在这里添加调用接口获取数据的代码  
                    //doSomething()  
                } catch (Exception e) {  
                    e.printStackTrace();  
                }  
                return true;  
            }  
            @Override  
            protected void onPostExecute(Boolean isSuccess) {  
                if (isSuccess) {  
                    // 加载成功  
                    setView();  
                    mHasLoadedOnce = true;  
                } else {  
                    // 加载失败  
                }  
                progressBar.setVisibility(View.GONE);  
            }  
        }.execute();  
    }  
  
    private void setView() {  
        // 根据索引加载不同视图  
        switch (mCurIndex) {  
            case FIRST_FRAGMENT:  
                contentText.setText("第一个");  
                break;  
  
            case SECOND_FRAGMENT:  
                contentText.setText("第二个");  
                break;  
  
            case THIRD_FRAGMENT:  
                contentText.setText("第三个");  
                break;  
        }  
    }  
  
    @Override  
    public void onDestroyView() {  
        super.onDestroyView();  
        Log.d("LiaBin", "onDestroyView: curIndex=" + mCurIndex);  
    }  
  
    @Override  
    public void onDestroy() {  
        super.onDestroy();  
        Log.d("LiaBin", "onDestroy: curIndex=" + mCurIndex);  
    }  
}  

  

ViewPager防止Fragment销毁以及取消Fragment的预加载的更多相关文章

  1. 防止ViewPager和Fragment结合使用时候的数据预加载

    不知道你们使用ViewPager和Fragment结合的时候发现一个问题没,如果你的每个Fragment都需要请求网络数据,并且你在请求网络数据的时候会加入进度对话框的加载显示效果,当你显示第一个Fr ...

  2. Android中ViewPager+Fragment取消(禁止)预加载延迟加载(懒加载)问题解决方案

    转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53205878本文出自[DylanAndroid的博客] Android中Vie ...

  3. ViewPager+Fragment取消预加载(延迟加载)(转)

    原文:http://www.2cto.com/kf/201501/368954.html 在项目中,都或多或少地使用的Tab布局,所以大都会用到ViewPager+Fragment,但是Fragmen ...

  4. ViewPager+Fragment取消预加载(延迟加载)

    在项目中,都或多或少地使用的Tab布局,所以大都会用到ViewPager+Fragment,但是Fragment有个不好或者太好的地方.例如你在ViewPager中添加了三个Fragment,当加载V ...

  5. android Viewpager取消预加载及Fragment方法的学习

    1.在使用ViewPager嵌套Fragment的时候,由于VIewPager的几个Adapter的设置来说,都会有一定的预加载.通过设置setOffscreenPageLimit(int numbe ...

  6. viewpager和fragment预加载的解决

    在使用Viewpager和fragment处理中会出现预加载的问题,最近看别人的代码,终于找到了一个很好的处理方法 能有效的解决预加载的问题,在fragment都继承一个重写setUserVisibl ...

  7. 巧力避免ViewPager的预加载数据,Tablayout+Fragment+viewPager

    问题描述 最近在进行一个项目的开发,其中使用到了Tablayout+Fragment+viewPager来搭建一个基本的框架,从而出现了设置数据适配器的时候,item的位置错乱问题.我打印log日志的 ...

  8. Fragment禁止预加载

    项目中经常会用到ViewPager+Fragment组合,然而,有一个很让人头疼的问题就是,去加载数据的时候由于ViewPager的内部机制所限制,所以它会默认至少预加载一个. 1.既然说是ViewP ...

  9. Fragment懒加载预加载

    1. 预加载viewpager.setOffscreenPageLimit(2);,默认是预加载1,可以结合懒加载使用. 如果希望进入viewpager,Fragment只加载一次,再次滑动不需加载( ...

随机推荐

  1. 设置win版vim启动项[利刃篇]

    vim默认启动设置有时不太好,想改一下,但是又不像windows其他软件那样,界面设置后就生效,这里需要手动设置文件才可以,那就赶紧设置一下吧! 修改文件为 vim安装目录/_vimrc 文件,或者在 ...

  2. 双系统Ubuntu无法访问Win10磁盘分区解决方法

    今天在linux下打开win的NTFS硬盘总是提示出错了,而且是全部的NTFS盘都出错,其中sda3错误显示如下: Error mounting /dev/sda3 at /media/struggl ...

  3. HoloLens开发手记 - 语音输入 Voice input

    语音是HoloLens三大重要输入形式之一.它允许你直接通过语言控制全息图像,而不用借助手势.你只要凝视全息图像然后说出语音命令即可.语音输入是自然的交互方式,它能够很好的改善复杂的交互,因为通过一条 ...

  4. 【xsy1596】旅行 期望+状压DP

    题目大意:有$m$个人要从城市$1$开始,依次游览城市$1$到$n$. 每一天,每一个游客有$p_i$的概率去下一个城市,和$1-p_i$的概率结束游览. 当游客到达城市$j$,他会得到$(1+\fr ...

  5. web自动化测试---测试中其他一些常用操作

    一些其他常用操作如下: 1.最大化浏览器窗口 driver.maximize_window() 2.后退 driver.back() 3.前进 driver.forward() 4.刷新操作 driv ...

  6. Java访问文件夹中文件的递归遍历代码Demo

    上代码: import java.io.File; /* * 需求:对指定目录进行所有内容的列出(包含子目录中的内容) * 也可以理解为 深度遍历. */ public class FindAllFi ...

  7. Hive ROW_NUMBER,RANK(),DENSE_RANK()

    准备数据 浙江,杭州,300 浙江,宁波,150 浙江,温州,200 浙江,嘉兴,100 江苏,南京,270 江苏,苏州,299 江苏,某市,200 江苏,某某市,100   创建表 CREATE t ...

  8. 从零开始学 Web 之 CSS(一)选择器

    大家好,这里是「 Daotin的梦呓 」从零开始学 Web 系列教程.此文首发于「 Daotin的梦呓 」公众号,欢迎大家订阅关注.在这里我会从 Web 前端零基础开始,一步步学习 Web 相关的知识 ...

  9. jfinal定时任务插件jfinal-quartz

    这个定时任务插件精确的时间可以到秒,使用方面跟jfinal-scheduler插件的使用方式差不多 Dreampie/jfinal-quartz https://github.com/Dreampie ...

  10. VM虚拟机扩展硬盘容量

    VM虚拟机扩展硬盘容量 第一步,关闭系统,给虚拟机硬盘增加空间. 第二步,启动系统.查看硬盘大小和分区情况. 第三步,分区. 第四步,格式化分区. 第五步,挂载. 第六步,开机自动挂载. 第一步: 当 ...