解决ViewPager和PagerAdapter中调用notifyDataSetChanged失效的问题


Google在Android 3.0SDK中推出的ViewPager控件很大程度上满足了开发者开发页面左右移动切换的功能,使用非常方便。但是使用中发现,在删除或者修改数据的时候,PagerAdapter无法像BaseAdapter那样仅通过notifyDataSetChanged方法通知刷新View。
最基本的方法:
针对于child view比较简单的情况(例如仅有TextView、ImageView等,没有ListView等展示数据的情况),可以在自己的Adapter中加入代码:

  1. @Override
  2. public int getItemPosition(Object object) {
  3. return POSITION_NONE;
  4. }
    @Override  
    public int getItemPosition(Object object) {  
        return POSITION_NONE;  
    }  
这样既可达到一般情况下要求的效果。
存在的问题:
这不是PagerAdapter中的Bug,通常情况下,调用notifyDataSetChanged方法会让ViewPager通过Adapter的getItemPosition方法查询一遍所有child view,这种情况下,所有child view位置均为POSITION_NONE,表示所有的child view都不存在,ViewPager会调用destroyItem方法销毁,并且重新生成,加大系统开销,并在一些复杂情况下导致逻辑问题。特别是对于只是希望更新child view内容的时候,造成了完全不必要的开销。
更有效地方法:
更为靠谱的方法是因地制宜,根据自己的需求来实现notifyDataSetChanged的功能,比如,在仅需要对某个View内容进行更新时,在instantiateItem()时,用View.setTag方法加入标志,在需要更新信息时,通过findViewWithTag的方法找到对应的View进行更新即可。


使用ViewPager做滑动切换图片的效果时,如果图片是从网络下载的,那么再子线程中下载完图片时我们会使用handler通知UI线程,然后UI线程就可以调用mViewPager.getAdapter().notifyDataSetChanged()进行页面的刷新,但是viewpager不同于listview,你会发现单纯的调用notifyDataSetChanged()并不能刷新页面。先说说Viewpager的刷新过程:在每次调用notifyDataSetChanged()时,都会激活getItemPosition(Object object)方法,该方法会遍历viewpager的所有item(据我debug的结果,只有当前页和其左右加起来共3页被遍历了,待确定),为每个item返回一个状态值(POSITION_NONE/POSITION_UNCHANGED),如果是none,那么该item会被destroyItem(ViewGroup container, int position, Object object)方法remove掉,然后重新加载,如果是unchanged,就不会重新加载,默认是unchanged,所以我国我们不重写getItemPosition(Object object),就无法看到刷新效果。解决方法有两种:
第一种网上比较容易查找到:重写PagerAdapter的getItemPosition(Object object)方法,使其返回POSITION_NONE

Java代码 复制代码 收藏代码
  1. @Override
  2. public int getItemPosition(Object object) {
  3. return POSITION_NONE;
  4. }
@Override
public int getItemPosition(Object object) {
	return POSITION_NONE;
}

这种方法的弊端大家都很容易看出来,我不需要刷新的项目也被重新加载了,浪费系统资源;
第二种更合理,当然相对前一种要再多做点事:思路是在instantiateItem时给每个view加上tag,然后在需要刷新页面时通过View.getTag()来判断是否是我们想要刷新的页面,只给当前页面返回POSITION_NONE。
Java代码 复制代码 收藏代码
  1. /**
  2. * DispImgAdapter.java
  3. */
  4. @Override
  5. public Object instantiateItem(ViewGroup container, int position) {
  6. iv = new ImageView(mContext);
  7. iv.setTag(position); // Add tag
  8. try {
  9. Bitmap bm = cacheImg2(position);
  10. iv.setImageBitmap(bm);
  11. } catch (OutOfMemoryError e) {
  12. e.printStackTrace();
  13. }
  14. ((ViewPager)container).addView(iv);
  15. return iv;
  16. }
  17. @Override
  18. public int getItemPosition(Object object) {
  19. View view = (View)object;
  20. int currentPage = ((DispImgActivity)mContext).getCurrentPagerIdx(); // Get current page index
  21. if(currentPage == (Integer)view.getTag()){
  22. return POSITION_NONE;
  23. }else{
  24. return POSITION_UNCHANGED;
  25. }
  26. // return POSITION_NONE;
  27. }
/**
 * 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便是。
Java代码 复制代码 收藏代码
  1. /**
  2. * DispImgActivity.java
  3. */
  4. // Get current page index
  5. mViewPager.setOnPageChangeListener(new OnPageChangeListener() {
  6. @Override
  7. public void onPageScrolled(int i, float f, int j) {
  8. }
  9. @Override
  10. public void onPageSelected(int position) {
  11. DispImgActivity.this.position = position;
  12. }
  13. @Override
  14. public void onPageScrollStateChanged(int i) {
  15. }
  16. });
  17. // Return current index to Adapter
  18. public int getCurrentPagerIdx() {
  19. return position;
  20. }
/**
 * 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;
}


PS:我的项目中还加入了图片下载进度条的功能,当我用第二种方法时,在一些比较极端的情况下会有一点问题,假设所有图片都需要从网上下载,在极快速滑动页面时,发现偶尔会出现异步下载到的图片并没有被刷新显示,在滑过几页重新回到该页时图片才被刷新了,这里涉及到的关键问题是【ViewPager的预加载机制+图片异步下载+getItemPosition中对Tag的判断】,我认为是这几种机制结合后再快速切换页面时造成的问题,由于项目工期的限制,没有去探索更完美的解决方法,反正图片也不是很多,我就采用了第一种方法来做,可以完美的实现我的功能。  




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值