viewPager+fragment如何刷新缓存fragment

时间:2023-03-09 02:32:44
viewPager+fragment如何刷新缓存fragment

最近在做一个项目,有一个功能是答题翻页。于是需要实现在这一页的时候就缓存下一页。

刚刚开始我是用

setOnPageChangeListener方法监听,滑到这一页的时候才刷新这一页:
 public void onPageSelected(int position)
{
ReadFragment fragment= (ReadFragment) fragmentArrayList.get(position);
fragment.refresh();
}

不过这样就只有滑动到这一页的时候才能用fragmentArrayList.get(position)获取当前页,用这种方法获取下一页的fragment就会报空指针。也就是说无法先缓存刷新下一页的内容。

到底怎么样才能获取得到下一页的fragment呢?

百度了一下好像说要在

FragmentPagerAdapter里面的instantiateItem()处理。于是我看了一下它的源代码:
@Override
public Object instantiateItem(ViewGroup container, int position) {
if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
} final long itemId = getItemId(position); // Do we already have this fragment?
String name = makeFragmentName(container.getId(), itemId);
Fragment fragment = mFragmentManager.findFragmentByTag(name);
if (fragment != null) {
if (DEBUG) Log.v(TAG, "Attaching item #" + itemId + ": f=" + fragment);
mCurTransaction.attach(fragment);
} else {
fragment = getItem(position);
if (DEBUG) Log.v(TAG, "Adding item #" + itemId + ": f=" + fragment);
mCurTransaction.add(container.getId(), fragment,
makeFragmentName(container.getId(), itemId));
}
if (fragment != mCurrentPrimaryItem) {
fragment.setMenuVisibility(false);
fragment.setUserVisibleHint(false);
} return fragment;
}

可以看出:

instantiateItem方法中并不是直接去List里面拿到Fragment,而是先从FragmentManager中通过Tag找对应的Fragment,如果可以找到就不会去List里面拿了,介于这种情况,我在adapter中加入了这个方法:

 public ReadFragment getFragment(int position)
{
String tag = getFragmentTag(mContainer.getId(),position);
ReadFragment fragment = (ReadFragment) fm.findFragmentByTag(tag);
return fragment; } /**
* 运用反射机制调用FragmentPagerAdapter的getFragmentTag的方法
* @param viewId
* @param index
* @return
*/
private String getFragmentTag(int viewId, int index)
{
try {
Class<FragmentPagerAdapter> cls = FragmentPagerAdapter.class;
Class<?>[] parameterTypes = { int.class, long.class };
Method method = cls.getDeclaredMethod("makeFragmentName",
parameterTypes);
method.setAccessible(true);
String tag = (String) method.invoke(this, viewId, index);
return tag;
} catch (Exception e) {
e.printStackTrace();
return "";
}
}

在onPageSelected里面调用getFragment(int position)方法获取下一页的内容,达到当选中这一页的时候就先缓存刷新下一页。

getFragment(int position)方法其实就是拿到缓存的fragment(这里的缓存是指viewpager缓存的fragment),不过就得先保证该fragment已经先在viewpager中缓存了,虽然内容还没有刷新(每个fragment里面显示的内容都是一样的),这样就不会报空指针了。

出现了一个问题,onPageSelected在viewPager展示第一页的时候是不会调用的,所以第一页的内容还是得另外刷新,无法在onPageSelected里面刷新

建立一个方法initData(),在里面单独刷新第一页的fragment。

由于viewPager展示第一页的时候不会调用onPageSelected,那也就导致了第一页和第二页的内容都无法先得到缓存(因为第二页是在第一页选中的时候就事先开始缓存的),所以第一页和第二页的内容都得在initData()里面单独刷新,其它的通过onPageSelected里面的方法来刷新。

到了这里总结一下思路:

刚刚进入界面的时候:刷新第一页,缓存第二页。

翻页时候:从第一页翻到第二页,执行onPageSelected()

              onPagerSelected里面调用方法getFragment(int position),获取到下一页即第三页的fragment,然后刷新缓存内容。

              从第二页翻到第三页:执行onPageSelected()

              onPagerSelected里面调用方法getFragment(int position),获取到下一页即第四页的fragment,然后刷新缓存内容。

结果又出现了又一个问题:从第一页翻到第二页的时候,闪退了,报空指针。

后来调试了一下发现getFragment方法得到的fragment为null,没道理,为啥,想到最后才发现原来是因为第三页的fragment在viewPager中没有缓存,而我们的getFragment(int position)是通过tag标记向缓存中拿的。

怎么才能让第三个fragment在viewPager中实现得到缓存呢?

默认的,viewpager在第一页的时候会缓存第二页,到了第二页的时候会缓存第一与第二页,实践证明,只有当第二页完全显示的时候,第三页才会得到缓存,而onPagerSelected在fragment滑到超过屏幕一半而且我们手指放开了才会调用,如果我们的手指没有放开是不会被调用的,当我们的手指放开,onPagerSelected被调用的时候,第三页还没有得到缓存。

怎么办?我又想到了

@Override public void onPageScrollStateChanged(int state) {}
本认为可以在里面判断state==2,即滑动停止的时候,才缓存刷新这一页,最后才发现还是有一样的问题 原来滑动停止指的是手指的滑动,即手指离开屏幕,而不是指改fragment的滑动,因此还是会出现上面的情况。

怎么办,不用怕:还有一个方法:readViewPager.setOffscreenPageLimit(2);

该方法可以给你答案。

这个方法可以设置viewPager当前页两边的缓存数目,如readViewPager.setOffscreenPageLimit(2);当前页左右各缓存2个。

viewPager默认的是readViewPager.setOffscreenPageLimit(1);

这样就OK了?,高兴太早了。滑到第五页的时候出问题了,是空白的。改为readViewPager.setOffscreenPageLimit(3);也一样。

(此时我总共只有4个fragment,我采用的是无限循环模式,实际的fragment有4个)而实际缓存的有5个fragment(当前页加左右两个),会不会是这里出问题。

关于怎么样设置无限循环,可以参考我另外一篇博客:http://www.cnblogs.com/tangZH/p/6516566.html

于是我把fragment改为5个,结果没问题了。

但是往回翻页的时候出问题了,翻几页后又出现了空白页,于是我改为6个fragment,才完全没问题。

晕。

还有往回翻页也需要刷新上一页的内容。