1、PagerAdapter 的工作流程
其实就是 PagerAdapter 中方法的执行顺序,来看看 Leo8573 的分析(个人感觉基本说到位了,所以直接拷过来了):
PagerAdapter 作为 ViewPager 的适配器,无论 ViewPager 有多少页,PagerAdapter 在初始化时也只初始化开始的2个 View,即调用2次instantiateItem 方法。而接下来每当 ViewPager 滑动时,PagerAdapter 都会调用 destroyItem 方法将距离该页2个步幅以上的那个 View 销毁,以此保证 PagerAdapter 最多只管辖3个 View,且当前 View 是3个中的中间一个,如果当前 View 缺少两边的 View,那么就 instantiateItem,如里有超过2个步幅的就 destroyItem。
简易图示:
滑动后,当前 View 变为3号 View,PagerAdapter 会 destroyItem 0号View,instantiateItem 5号 View,所以 PagerAdapter 管辖2、3、4三个 View。(参考自:关于ViewPager的数据更新问题小结)
总结一下: Viewpager 的刷新过程是这样的,在每次调用 PagerAdapter 的 notifyDataSetChanged() 方法时,都会激活 getItemPosition(Object object) 方法,该方法会遍历 ViewPager 的所有 Item(由缓存的 Item 数量决定,默认为当前页和其左右加起来共3页,这个可以自行设定,但是至少会缓存2页),为每个 Item 返回一个状态值(POSITION_NONE/POSITION_UNCHANGED),如果是 POSITION_NONE,那么该 Item 会被 destroyItem(ViewGroup container, int position, Object object) 方法 remove 掉,然后重新加载,如果是 POSITION_UNCHANGED,就不会重新加载,默认是 POSITION_UNCHANGED,所以如果不重写 getItemPosition(Object object),修改返回值,就无法看到 notifyDataSetChanged() 的刷新效果。
2、刷新数据的方法
第一种方法:重写PagerAdapter的getItemPosition(Object object)方法,使其返回POSITION_NONE
@Override public int getItemPosition(Object object) { return POSITION_NONE; }
这种方法的弊端大家都很容易看出来,我不需要刷新的项目也被重新加载了,浪费系统资源;
第二种更合理,当然相对前一种要再多做点事:思路是在instantiateItem时给每个view加上tag,然后在需要刷新页面时通过View.getTag()来判断是否是我们想要刷新的页面,只给当前页面返回POSITION_NONE。
/** * DispImgAdapter.java */ @Override public Object instantiateItem(ViewGroup container, int position) { iv = new ImageView(mContext); iv.setTag(position); // Add tag try { Bitmap bm = cacheImg2(position); iv.setImageBitmap(bm); } catch (OutOfMemoryError e) { e.printStackTrace(); } ((ViewPager)container).addView(iv); return iv; } @Override public int getItemPosition(Object object) { View view = (View)object; int currentPage = ((DispImgActivity)mContext).getCurrentPagerIdx(); // Get current page index if(currentPage == (Integer)view.getTag()){ return POSITION_NONE; }else{ return POSITION_UNCHANGED; } // return POSITION_NONE; }
关键的currentPageIdx则需要在Activity中获取,如果你的Adapter是Activity的内部类,那么只要把index写成全局变量就可以在adapter中使用了,如果是单独的两个类,那么你就自己提供一个接口,将index传给Adapter便是。
/** * DispImgActivity.java */ // Get current page index mViewPager.setOnPageChangeListener(new OnPageChangeListener() { @Override public void onPageScrolled(int i, float f, int j) { } @Override public void onPageSelected(int position) { DispImgActivity.this.position = position; } @Override public void onPageScrollStateChanged(int i) { } }); // Return current index to Adapter public int getCurrentPagerIdx() { return position; }