博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
带你走过ViewPager不刷新界面的坑
阅读量:5812 次
发布时间:2019-06-18

本文共 5439 字,大约阅读时间需要 18 分钟。

最近在项目中用到ViewPager+FragmentPagerAdapter的方式来做界面,其中当adapter的数据源数据更新时,调用adapter.notifyDataSetChanged()更新数据,发现ViewPager并没有更新,还是原来的数据。

参考了别人的文章以及部分解决的的方法,加上自己的理解,拿出了下面这套解决方案。

目录:

  1. 问题展示
  2. 解决方案
  3. 问题追究

问题展示

国际惯例,先上问题图(图一)以及正常图(图二)。

解决方案

  1. 在数据源更新的前面加入以下代码
if (viewPager.getAdapter() != null) {    FragmentManager fm = getSupportFragmentManager();    FragmentTransaction ft = fm.beginTransaction();    List
fragments = fm.getFragments(); if(fragments != null && fragments.size() >0){ for (int i = 0; i < fragments.size(); i++) { ft.remove(fragments.get(i)); } } ft.commit();}复制代码
  1. 在你的adapter类中加入以下代码
private int mChildCount = 0;@Overridepublic void notifyDataSetChanged() {    // 重写这个方法,取到子Fragment的数量,用于下面的判断,以执行多少次刷新        mChildCount = getCount();    super.notifyDataSetChanged();}@Overridepublic int getItemPosition(Object object) {    if ( mChildCount > 0) {        // 这里利用判断执行若干次不缓存,刷新        mChildCount --;        // 返回这个是强制ViewPager不缓存,每次滑动都刷新视图        return POSITION_NONE;    }    // 这个则是缓存不刷新视图    return super.getItemPosition(object);}复制代码
较完整代码一览

初始化数据

viewPager= (ViewPager) findViewById(R.id.pager);mList=new ArrayList
();for (int i=1;i<4;i++){ Bundle bundle=new Bundle(); bundle.putString("text","第"+i+"页"); MyFragment myFragment=new MyFragment(); myFragment.setArguments(bundle); mList.add(myFragment);}adapter=new MyAdapter(getSupportFragmentManager());viewPager.setAdapter(adapter);复制代码

自定义adapter

public class MyAdapter extends FragmentPagerAdapter{    public MyAdapter(FragmentManager fm) {        super(fm);    }    @Override    public Fragment getItem(int position) {        return mList.get(position);    }    @Override    public int getCount() {        return mList.size();    }    // start    // 可以删除这段代码看看,数据源更新而viewpager不更新的情况    private int mChildCount = 0;    @Override    public void notifyDataSetChanged() {        // 重写这个方法,取到子Fragment的数量,用于下面的判断,以执行多少次刷新        mChildCount = getCount();        super.notifyDataSetChanged();    }    @Override    public int getItemPosition(Object object) {        if ( mChildCount > 0) {            // 这里利用判断执行若干次不缓存,刷新            mChildCount --;            // 返回这个是强制ViewPager不缓存,每次滑动都刷新视图            return POSITION_NONE;        }        // 这个则是缓存不刷新视图        return super.getItemPosition(object);    }    // end}复制代码

更新数据源的方法

public void update(){    // start    // 可以删除这段代码看看,数据源更新而viewpager不更新的情况    // 在数据源更新前增加的代码,将上一次数据源的fragment对象从FragmentManager中删除    if (viewPager.getAdapter() != null) {        FragmentManager fm = getSupportFragmentManager();        FragmentTransaction ft = fm.beginTransaction();        List
fragments = fm.getFragments(); if(fragments != null && fragments.size() >0){ for (int i = 0; i < fragments.size(); i++) { ft.remove(fragments.get(i)); } } ft.commit(); } // End mList.clear(); for (int i=4;i<7;i++){ Bundle bundle=new Bundle(); bundle.putString("text","第"+i+"页"); MyFragment myFragment=new MyFragment(); myFragment.setArguments(bundle); mList.add(myFragment); } // 重写adapter的notifyDataChanged方法 adapter.notifyDataSetChanged();}复制代码
demo源码下载

可根据注释删除对应的代码,体验有问题以及正常的情况。

问题追究

首先来理解两个adapter,都是继承与pageradapter

  1. FragmentPagerAdapter:该类更专注于每一页均为 Fragment 的情况。该类内的每一个生成的 Fragment 都将保存在内存之中,因此适用于那些相对静态的页,数量也比较少的那种;如果需要处理有很多页,并且数据动态性较大、占用内存较多的情况,应该使用FragmentStatePagerAdapter
  2. FragmentStatePagerAdapter:和 FragmentPagerAdapter不一样的是,正如其类名中的 'State' 所表明的含义一样,该 PagerAdapter 的实现将只保留当前页面,当页面离开视线后,就会被消除,释放其资源;而在页面需要显示时,生成新的页面(就像 ListView 的实现一样)。这么实现的好处就是当拥有大量的页面时,不必在内存中占用大量的内存。
  3. 这两个adapter最大的不同在于instantiateItem()这个方法

接下来看看adapter里面getItemPosition这个方法 可以返回的值为POSITION_UNCHANGEDPOSITION_NONE这两个值。 而默认都是返回POSITION_UNCHANGED 这个返回值会在adapter的instantiateItem()方法里进行判断: POSITION_UNCHANGED不重新加载item POSITION_NONE要求重新加载item

而网上的一些解决方案是直接复写FragmentPagerAdaptergetItemPosition返回POSITION_NONE,这样做及违反了FragmentPagerAdapter的设计原则(保存在内存,加载更快等)也没有解决今天这个坑,一样是界面没有刷新的。

继续说下去

假如返回POSITION_NONE要求从新加载Item,ViewPager会首先去FragmentManager里面去查找有没有相关的fragment如果有就直接使用如果没有才会触发FragmentPageadAptergetItem方法获取一个fragment。所以你更新的fragmentList集合是没有作用的,还要清除FragmentManager里面缓存的fragment

这样今天的解决方案思路救出来了:

  1. 复写notifyDataSetChanged
@Overridepublic void notifyDataSetChanged() {    // 重写这个方法,取到子Fragment的数量,用于下面的判断,以执行多少次刷新    mChildCount = getCount();    super.notifyDataSetChanged();}复制代码
  1. 复写getItemPosition,根据mChildCount判断是返回POSITION_UNCHANGED还是itemPOSITION_NONE
@Overridepublic int getItemPosition(Object object) {    if ( mChildCount > 0) {        // 这里利用判断执行若干次不缓存,刷新        mChildCount --;        // 返回这个是itemPOSITION_NONE        return POSITION_NONE;    }    // 这个则是POSITION_UNCHANGED    return super.getItemPosition(object);}复制代码
  1. notifyDataSetChanged之前对FragmentManager进行相应的删除操作。
if (viewPager.getAdapter() != null) {    FragmentManager fm = getSupportFragmentManager();    FragmentTransaction ft = fm.beginTransaction();    List
fragments = fm.getFragments(); if(fragments != null && fragments.size() >0){ for (int i = 0; i < fragments.size(); i++) { ft.remove(fragments.get(i)); } } ft.commit();}复制代码
  1. 这样就会在notifyDataSetChanged的时候刷新视图,在平时滑动等情况使用缓存视图,既保留了FragmentPagerAdapter的特点,又解决了今天的坑。

到此,今天的坑又总算是跨过去了,如果有帮组到你,欢迎关注和 本文参考自:

  • http://www.cnblogs.com/lianghui66/p/3607091.html
  • http://blog.sina.com.cn/s/blog_783ede03010173b4.html

2016年9月7日 00:35:49

你可能感兴趣的文章
Android文件下载之进度检测
查看>>
重构第10天:提取方法(Extract Method)
查看>>
吐血整理 Delphi系列书籍 118本(全)
查看>>
Android Fragment使用(四) Toolbar使用及Fragment中的Toolbar处理
查看>>
解决pycharm在ubuntu下搜狗输入法一直固定在左下角的问题
查看>>
“Info.plist” couldn’t be removed
查看>>
Linux创建系统用户
查看>>
多线程day01
查看>>
JSON path
查看>>
Win8 Metro(C#)数字图像处理--2.43图像马赛克效果算法
查看>>
动画库NineOldAndroids
查看>>
react-native 模仿原生 实现下拉刷新/上拉加载更多(RefreshListView)
查看>>
大数据开发实战:Hadoop数据仓库开发实战
查看>>
Spring Boot 2中对于CORS跨域访问的快速支持
查看>>
MySQL出现Access denied for user ‘root’@’localhost’ (using password:YES)
查看>>
matlab fread
查看>>
通过Roslyn构建自己的C#脚本(更新版)(转)
查看>>
红黑树
查看>>
mybatis08
查看>>
01 awk工具的使用
查看>>