为何调用 FragmentPagerAdapter.notifyDataSetChanged() 并不能更新其 Fragment?【转载】

 

转载自:http://www.cnblogs.com/dancefire/archive/2013/01/02/why-notifyDataSetChanged-does-not-work.htmlhtml

 

在一个 Android 应用中,我使用 FragmentPagerAdapter 来处理多 Fragment 页面的横向滑动。不过我碰到了一个问题,即当 Fragment 对应的数据集发生改变时,我但愿可以经过调用 mAdapter.notifyDataSetChanged() 来触发 Fragment 页面使用新的数据调整或从新生成其内容,但是当我调用 notifyDataSetChanged() 后,发现什么都没发生。java

   

搜索以后发现不止我一我的碰到这个问题,你们给出的解决办法五花八门,有些确实解决了问题,可是我总感受问题没搞清楚。因而我决定搞明白这个问题究竟是怎么回事,以及正确的用法到底如何。要搞明白这个问题,仅仅阅读文档并不足够,还须要阅读相关几个类的相关方法的实现,搞懂其设计意图。下面就是经过阅读源代码搞明白的内容。android

   

   

ViewPagerapp

 

ViewPager 如其名所述,是负责翻页的一个 View。准确说是一个 ViewGroup,包含多个 View 页,在手指横向滑动屏幕时,其负责对 View 进行切换。为了生成这些 View 页,须要提供一个 PagerAdapter 来进行和数据绑定以及生成最终的 View 页。ide

   

    • ViewPager 经过 setAdapter() 来创建与 PagerAdapter 的联系。这个联系是双向的,一方面,ViewPager 会拥有 PagerAdapter 对象,从而能够在须要时调用 PagerAdapter 的方法;另外一方面,ViewPager 会在 setAdapter() 调用 PagerAdapter registerDataSetObserver() 方法,注册一个本身生成的 PagerObserver 对象,从而在 PagerAdapter 有所须要时(如 notifyDataSetChanged()  notifyDataSetInvalidated() 时),能够调用 Observer onChanged() onInvalidated() 方法,从而实现 PagerAdapter ViewPager 方向发送信息。
    •  PagerObserver.onChanged(),以及 PagerObserver.onInvalide() 中被调用。所以当 PagerAdapter.notifyDataSetChanged() 被触发时,ViewPager.dataSetChanged() 也能够被触发。该函数将使用 getItemPosition() 的返回值来进行判断,若是为 POSITION_UNCHANGED,则什么都不作;若是为 POSITION_NONE,则调用 PagerAdapter.destroyItem() 来去掉该对象,并设置为须要刷新 (needPopulate = true) 以便触发 PagerAdapter.instantiateItem() 来生成新的对象。

   

PagerAdapter函数

   

PageAdapter  ViewPager 的支持者,ViewPager 将调用它来取得所需显示的页,而 PageAdapter 也会在数据变化时,通知 ViewPager。这个类也是FragmentPagerAdapter 以及 FragmentStatePagerAdapter 的基类。若是继承自该类,至少须要实现 instantiateItem(), destroyItem(), getCount() 以及 isViewFromObject()ui

   

    • 该函数用以返回给定对象的位置,给定对象是由 instantiateItem() 的返回值。
    • ViewPager.dataSetChanged() 中将对该函数的返回值进行判断,以决定是否最终触发 PagerAdapter.instantiateItem() 函数。
    • PagerAdapter 中的实现是直接传回 POSITION_UNCHANGED。若是该函数不被重载,则会一直返回 POSITION_UNCHANGED,从而致使 ViewPager.dataSetChanged() 被调用时,认为没必要触发 PagerAdapter.instantiateItem()。不少人由于没有重载该函数,而致使调用
       PagerAdapter.notifyDataSetChanged()
      后,什么都没有发生。
    • 在每次 ViewPager 须要一个用以显示的 Object 的时候,该函数都会被 ViewPager.addNewItem() 调用。
    • 在数据集发生变化的时候,通常 Activity 会调用 PagerAdapter.notifyDataSetChanged(),以通知 PagerAdapter,而 PagerAdapter 则会通知在本身这里注册过的全部 DataSetObserver。其中之一就是在 ViewPager.setAdapter() 中注册过的 PageObserverPageObserver 则进而调用 ViewPager.dataSetChanged(),从而致使 ViewPager 开始触发更新其内含 View 的操做。

   

FragmentPagerAdaptergoogle

   

FragmentPagerAdapter 继承自 PagerAdapter。相比通用的 PagerAdapter,该类更专一于每一页均为 Fragment 的状况。如文档所述,该类内的每个生成的 Fragment 都将保存在内存之中,所以适用于那些相对静态的页,数量也比较少的那种;若是须要处理有不少页,而且数据动态性较大、占用内存较多的状况,应该使用FragmentStatePagerAdapterFragmentPagerAdapter 重载实现了几个必须的函数,所以来自 PagerAdapter 的函数,咱们只须要实现 getCount(),便可。且,因为 FragmentPagerAdapter.instantiateItem() 的实现中,调用了一个新增的虚函数 getItem(),所以,咱们还至少须要实现一个 getItem()。所以,整体上来讲,相对于继承自 PagerAdapter,更方便一些。spa

   

    • 该类中新增的一个虚函数。函数的目的为生成新的 Fragment 对象。重载该函数时须要注意这一点。在须要时,该函数将被 instantiateItem() 所调用。
    • 若是须要向 Fragment 对象传递相对静态的数据时,咱们通常经过 Fragment.setArguments() 来进行,这部分代码应当放到 getItem()。它们只会在新生成 Fragment 对象时执行一遍。
    • 若是须要在生成 Fragment 对象后,将数据集里面一些动态的数据传递给该 Fragment,那么,这部分代码不适合放到 getItem() 中。由于当数据集发生变化时,每每对应的 Fragment 已经生成,若是传递数据部分代码放到了 getItem() 中,这部分代码将不会被调用。这也是为何不少人发现调用 PagerAdapter.notifyDataSetChanged() 后,getItem() 没有被调用的一个缘由。
    • 函数中判断一下要生成的 Fragment 是否已经生成过了,若是生成过了,就使用旧的,旧的将被 Fragment.attach();若是没有,就调用 getItem() 生成一个新的新的对象将被 FragmentTransation.add()
    • FragmentPagerAdapter 会将全部生成的 Fragment 对象经过 FragmentManager 保存起来备用,之后须要该 Fragment 时,都会从 FragmentManager 读取,而不会再次调用 getItem() 方法
    • 若是须要在生成 Fragment 对象后,将数据集中的一些数据传递给该 Fragment,这部分代码应该放到这个函数的重载里。在咱们继承的子类中,重载该函数,并调用 FragmentPagerAdapter.instantiateItem() 取得该函数返回 Fragment 对象,而后,咱们该 Fragment 对象中对应的方法,将数据传递过去,而后返回该对象。
    • 不然,若是将这部分传递数据的代码放到 getItem()中,在 PagerAdapter.notifyDataSetChanged() 后,这部分数据设置代码将不会被调用。
    • 该函数被调用后,会对 Fragment 进行 FragmentTransaction.detach()。这里不是 remove(),只是 detach(),所以 Fragment 还在 FragmentManager 管理中,Fragment 所占用的资源不会被释放。

   

FragmentStatePagerAdapter设计

   

   

FragmentStatePagerAdapter 和前面的 FragmentPagerAdapter 同样,是继承子 PagerAdapter。可是,和 FragmentPagerAdapter 不同的是,正如其类名中的 'State' 所代表的含义同样,该 PagerAdapter 的实现将只保留当前页面,当页面离开视线后,就会被消除,释放其资源;而在页面须要显示时,生成新的页面(就像 ListView 的实现同样)。这么实现的好处就是当拥有大量的页面时,没必要在内存中占用大量的内存。

   

    • 一个该类中新增的虚函数。
    • 函数的目的为生成新的 Fragment 对象。
    • Fragment.setArguments() 这种只会在新建 Fragment 时执行一次的参数传递代码,能够放在这里。
    • 因为 FragmentStatePagerAdapter.instantiateItem() 在大多数状况下,都将调用 getItem() 来生成新的对象,所以若是在该函数中放置与数据集相关的 setter 代码,基本上均可以在 instantiateItem() 被调用时执行,但这和设计意图不符。毕竟还有部分可能是不会调用 getItem() 的。所以这部分代码应该放到 instantiateItem() 中。
    • 除非碰到 FragmentManager 恰好从 SavedState 中恢复了对应的 Fragment 的状况外,该函数将会调用 getItem() 函数,生成新的 Fragment 对象。新的对象将被 FragmentTransaction.add()
    • FragmentStatePagerAdapter 就是经过这种方式,每次都建立一个新的 Fragment,而在不用后就马上释放其资源,来达到节省内存占用的目的的。

   

讨论

   

以前看到一些解决办法,有的认为这是一个 bug,应该被修复;有的建议不用 FragmentPagerAdapter,而改用 FragmentStatePagerAdapter,而且重载 getItemPosition() 并返回 POSITION_NONE,以触发销毁对象以及重建对象。从上面的分析中看,后者给出的建议确实能够达到调用 notifyDataSetChanged() 后,Fragment 被以新的参数从新创建的效果。

   

可是问题在于,若是咱们只能这么解决这个问题,岂不是 FragmentPagerAdapter 就用不上了?最关键的是,两者对应的状况不一样。对于页面相对较少的状况,我仍旧但愿可以将生成的 Fragment 保存在内存中,在须要显示的时候直接调用,而不要产生生成、销毁对象的额外的开销,这样效率更高。这种状况下,选择 FragmentPagerAdapter 是更适合,不加考虑的选择 FragmentStatePagerAdapter 是不合适的。咱们不可以因噎废食。

   

所以,对于 FragmentPagerAdapter 的解决方案就是,分别重载 getItem() 以及 instantiateItem() 对象。getItem() 只用于生成新的与数据无关的 Fragment;而 instantiateItem() 函数则先调用父类中的 instantiateItem() 取得所对应的 Fragment 对象,而后,根据对应的数据,调用该对象对应的方法进行数据设置。

   

固然,不要忘记重载 getItemPosition() 函数,返回 POSITION_NONE,这个两个类的解决方案都须要的。两者不一样之处在于,FragmentStatePagerAdapter 在会在因 POSITION_NONE 触发调用的 destroyItem() 中真正的释放资源,从新创建一个新的 Fragment;而 FragmentPagerAdapter 仅仅会在 destroyItem() detach 这个 Fragment,在 instantiateItem() 时会使用旧的 Fragment,并触发 attach,所以没有释放资源及重建的过程。

   

这样,当 notifyDataSetChanged() 被调用后,会最终触发 instantiateItem(),而无论 getItem() 是否被调用,咱们都在重载的 instantiateItem() 函数中已经将所须要的数据传递给了相应的 Fragment。在 Fragment 接下来的 onCreateView(), onStart() 以及 onResume() 的事件中,它能够正确的读取新的数据,Fragment 被成功复用了。

   

这里须要注意一个问题,在 Fragment 没有被添加到 FragmentManager 以前,咱们能够经过 Fragment.setArguments() 来设置参数,并在 Fragment 中,使用 getArguments() 来取得参数。这是经常使用的参数传递方式。可是这种方式对于咱们说的状况不适用。由于这种数据传递方式只可能用一次,在 Fragment 被添加到 FragmentManager 后,一旦被使用,咱们再次调用 setArguments() 将会致使 java.lang.IllegalStateException: Fragment already active 异常。所以,咱们这里的参数传递方式选择是,在继承的 Fragment 子类中,新增几个 setter,而后经过这些 setter 将数据传递过去。反向也是相似。相关信息能够参考 [5]。哦,这些 setter 中要注意不要操做那些 View,这些 View 只有在 onCreateView() 事件后才能够操做。

   

针对 FragmentPagerAdapter 的解决办法以下列代码所示:

   

   

1 @Override

2 public Fragment getItem(int position) {

3 MyFragment f = new MyFragment();

4 return f;

5 }

6

7 @Override

8 public Object instantiateItem(ViewGroup container, int position) {

9 MyFragment f = (MyFragment) super.instantiateItem(container, position);

10 String title = mList.get(position);

11 f.setTitle(title);

12 return f;

13 }

14

15 @Override

16 public int getItemPosition(Object object) {

17 return PagerAdapter.POSITION_NONE;

18 }

   

   

参考

   

Android 文档:

[1] http://developer.android.com/reference/android/support/v4/view/PagerAdapter.html

[2] http://developer.android.com/reference/android/support/v4/app/FragmentPagerAdapter.html

[3] http://developer.android.com/reference/android/support/v4/app/FragmentStatePagerAdapter.html

[4] http://developer.android.com/reference/android/support/v4/view/ViewPager.html

[5] http://developer.android.com/guide/components/fragments.html#CommunicatingWithActivity

   

Android 源代码:

[6] http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.1.1_r1/android/support/v4/view/PagerAdapter.java#PagerAdapter

[7] http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.1.1_r1/android/support/v4/app/FragmentPagerAdapter.java#FragmentPagerAdapter

[8] http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.1.1_r1/android/support/v4/app/FragmentStatePagerAdapter.java#FragmentStatePagerAdapter

[9] http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.1.1_r1/android/support/v4/view/ViewPager.java#ViewPager

   

Android Issue List:

[10] http://code.google.com/p/android/issues/detail?id=19001

相关文章
相关标签/搜索