Android开发之ListView实现Item局部刷新

对于android中的ListView刷新机制,大多数的程序员都是很熟悉的,修改或者添加adapter中的数据源之后,然后调用notifyDataSetChanged()刷新ListView。在这种模式下,我们会在getView中,根据不同的数据源,让控件显示不同的内容。这种模式是最常见的刷新模式,当我们来回滑动ListView的时候,调用adapter的getView方法,然后listview对adapter返回的View进行绘制。这种模式下,View的显示内容或状态都记录在adapter里面的数据源中,listview的更新频率不频繁,它随着数据源的变化而更新。

  但是我们小编在做公司项目的时候,有个下载模块,因为可能同时下载好几个数据,所以用的listview展示所有正在下载的内容。因为下载进度要实时更新,所以要不停的调用notifyDateSetChanged刷新数据。这样会不停的重新绘制整个listview的界面,性能开销非常大。而且如果每个item有图片的话,每个item的图片都需要重新加载,就算图片做了内存缓存,刷新一下图片也会闪一下,不停的刷新就会导致各个item的图片不停的闪,体验一点都不好。

  那么对于上面问题,有没有解决办法呢?当然是有的。我们可以针对某一个item进行局部更新,而不影响其它没有修改的item。那么具体如何实现的呢?我们看下面的代码。

 private void updateView(int itemIndex) {
   //得到第一个可显示控件的位置,
   int visiblePosition = mListView.getFirstVisiblePosition();
   //只有当要更新的view在可见的位置时才更新,不可见时,跳过不更新
   if (itemIndex - visiblePosition >= ) {
    //得到要更新的item的view
    View view = mListView.getChildAt(itemIndex - visiblePosition);
    //调用adapter更新界面
    mAdapter.updateView(view, itemIndex);
   }
  }

  这个函数主要是根据传入的itemIndex来获取第itemIndex的数据所显示的view。itemIndex就是要修改的数据再List集合中的位置,比如我这里下载进度有更新,发了一个广播这里接收到了,需要修改该下载内容的进度条,广播接收器可以这么写:

 @Override
   public void onReceive(Context context, Intent intent) {
    AppContent appContent = intent.getParcelableExtra("appContent");
    if(appContent == null) return;
    int itemIndex = ;
    for(AppContent appContent : mList) {
     if(appContent.getUrl().equals(appContent.getUrl())) {
      itemIndex = mList.indexOf(appContent);
      appContent.setDownloadPercent(appContent.getDownloadPercent());
      break;
     }
    }
    updateView(itemIndex);
   }

  下面看Adapter的具体代码:

 public class AppContentAdapter extends BaseAdapter{
  private List<AppContent> mDates = null;
  private Context mContext;
  public AppContentAdapter(Context context) {
   this.mContext = context;
  }
  @Override
  public int getCount() {
   return mDates.size();
  }
  @Override
  public Object getItem(int position) {
   return mDates.get(position);
  }
  @Override
  public long getItemId(int position) {
   return position;
  }
  public void setDates(List<AppContent> mDates) {
   this.mDates = mDates;
  }
  @Override
  public View getView(int position, View convertView, ViewGroup parent) {
   ViewHolder holder = null;
   if (convertView == null) {
    holder = new ViewHolder();
    convertView = LayoutInflater.from(mContext).inflate(
      R.layout.listitem_download, null);
    holder.statusIcon = (DownloadPercentView) convertView.findViewById(R.id.status_icon);
    holder.name = (TextView) convertView.findViewById(R.id.name);
    holder.downloadPercent = (TextView) convertView.findViewById(R.id.download_percent);
    holder.progressBar = (ProgressBar) convertView.findViewById(R.id.progressbar);
    convertView.setTag(holder);
   } else {
    holder = (ViewHolder) convertView.getTag();
   }
   setData(holder, position);
   return convertView;
  }
  /**
  * 设置viewHolder的数据
  * @param holder
  * @param itemIndex
  */
  private void setData(ViewHolder holder, int itemIndex) {
   AppContent appContent = mDates.get(itemIndex);
   holder.name.setText(appContent.getName());
   holder.progressBar.setProgress(appContent.getDownloadPercent());
   setIconByStatus(holder.statusIcon, appContent.getStatus());
   if(appContent.getStatus() == AppContent.Status.PENDING) {
    holder.downloadPercent.setVisibility(View.INVISIBLE);
   } else {
    holder.downloadPercent.setVisibility(View.VISIBLE);
    holder.statusIcon.setProgress(appContent.getDownloadPercent());
    holder.downloadPercent.setText("下载进度:" + appContent.getDownloadPercent() + "%");
   }
  }
  /**
  * 局部刷新
  * @param view
  * @param itemIndex
  */
  public void updateView(View view, int itemIndex) {
   if(view == null) {
    return;
   }
   //从view中取得holder
   ViewHolder holder = (ViewHolder) view.getTag();
   holder.statusIcon = (DownloadPercentView) view.findViewById(R.id.status_icon);
   holder.name = (TextView) view.findViewById(R.id.name);
   holder.downloadPercent = (TextView) view.findViewById(R.id.download_percent);
   holder.progressBar = (ProgressBar) view.findViewById(R.id.progressbar);
   setData(holder, itemIndex);
  }
  /**
  * 根据状态设置图标
  * @param downloadPercentView
  * @param status
  */
  private void setIconByStatus(DownloadPercentView downloadPercentView, AppContent.Status status) {
   downloadPercentView.setVisibility(View.VISIBLE);
   if(status == AppContent.Status.PENDING) {
    downloadPercentView.setStatus(DownloadPercentView.STATUS_PEDDING);
   }
   if(status == AppContent.Status.DOWNLOADING) {
    downloadPercentView.setStatus(DownloadPercentView.STATUS_DOWNLOADING);
   }
   if(status == AppContent.Status.WAITING) {
    downloadPercentView.setStatus(DownloadPercentView.STATUS_WAITING);
   }
   if(status == AppContent.Status.PAUSED) {
    downloadPercentView.setStatus(DownloadPercentView.STATUS_PAUSED);
   }
   if(status == AppContent.Status.FINISHED) {
    downloadPercentView.setStatus(DownloadPercentView.STATUS_FINISHED);
   }
  }
  private class ViewHolder {
   private DownloadPercentView statusIcon;
   private TextView name;
   private TextView downloadPercent;
   private ProgressBar progressBar;
  }
 }

以上内容是关于Android开发之ListView实现Item局部刷新的全部内容,希望对大家有用,更多有关listview局部刷新问题,请登录我们官网查询,谢谢!

(0)

相关推荐

  • Android使用ListView批量删除item的方法

    本文实例讲述了Android使用ListView批量删除item的方法.分享给大家供大家参考,具体如下: 利用CheckBox选中一个或多个item,最后批量删除它们. 程序运行效果图如下:   下面开始上码: (代码已更正,已解决滚动时紊乱的问题) package com.test.adapter; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;

  • Android 中ListView setOnItemClickListener点击无效原因分析

    前言 最近在做项目的过程中,在使用listview的时候遇到了设置item监听事件的时候在没有回调onItemClick 方法的问题.我的情况是在item中有一个Button按钮.所以不会回调.上百度找到了解决办法有两种,如下: 1.在checkbox.button对应的view处加android:focusable="false" 复制代码 代码如下: android:clickable="false" android:focusableInTouchMode=&

  • Android ListView的item中嵌套ScrollView的解决办法

    前沿:有时候,listview 的item要显示的字段比较多,考虑到显示问题,item外面不得不嵌套ScrollView来实现,于是问题来了,当listview需要做点击事件时,由于ScrollView的嵌套使用,拦截了listvew点击事件:只好重写listview来实现了. /** * * @author 作者:易皇星 * * @da2016年10月24日 时间: * * @toTODO 类描述: 解决 ListView中嵌套ScrollView,ScrollView拦截ListView的I

  • Android中Listview点击item不变颜色及设置listselector 无效的解决方案

    这是同一个问题,Listview中点击item是会变颜色的,因为listview设置了默认的listselector,有一个默认的颜色,同理如果点击没颜色变化我们怎么设置listselector也不会变颜色的. 但是在我们的开发过程中,我们可能会碰到这样的问题listview点击不变颜色,总结了一下大概有这几种原因: 1.item的layout设置background颜色值,去掉背景颜色即可 2.listview中listselector属性的效果被覆盖了,比如列表的Item为一个占满单元格的I

  • Android ListView的item背景色设置和item点击无响应的解决方法

    下面讲解以下在使用listview时最常见的几个问题.1.如何改变item的背景色和按下颜色 listview默认情况下,item的背景色是黑色,在用户点击时是黄色的.如果需要修改为自定义的背景颜色,一般情况下有三种方法: 1)设置listSelector 2)在布局文件中设置item的background 3)在adapter的getview中设置 这三种方法都能达到改变item默认的背景色和按下颜色,下面来分别讲解,但是在这之前需要先写好selector.xml文件; 复制代码 代码如下:

  • Android实现带有边框的ListView和item的方法

    本文实例讲述了Android实现带有边框的ListView和item的方法.分享给大家供大家参考,具体如下: 想为ListView和item四周添加边框有两种方法: 1.贴一张带有边框效果的背景图 2.自定义Draw的方法 第一种方法较第二种方法更耗系统资源,但是用法简单,只需要一张图设置为相应控件的背景即可,而第二种灵活性好些. 这次是实现带有边框的ListView和item,为此写个简单Demo 学习学习 先看下Demo运行效果吧 下面是主要代码,主要是用到Canvas.drawLine(.

  • Android中ListView Item布局优化技巧

    本文实例讲述了Android中ListView Item布局优化技巧.分享给大家供大家参考,具体如下: 之前一直都不知道ListView有多种布局的优化方法,只能通过隐藏来实现,自己也知道效率肯定是很低的,但是也不知道有什么方法,这些天又查了一些资料,然后知道 其实google早就帮我们想好了优化方案了. 假设你的ListView Item有三种布局样式的可能:就比如很简单的显示一行字,要靠左,居中,靠右. 这时我们就可以在BaseAdapter里面重写两个方法: private static

  • Android ListView的OnItemClickListener详解

    我们在使用ListView的时候,一般都会为ListView添加一个响应事件android.widget.AdapterView.OnItemClickListener.本文主要在于对OnItemClickListener的position和id参数做详细的解释,我相信有些人在这上面走了些弯路. 先来看一下官方的文档 position The position of the view in the adapter. id The row id of the item that was click

  • Android 中ListView的Item点击事件失效的快速解决方法

    在平常的开发过程中,我们的ListView可能不只是简单的显示下文本或者按钮,更多的是显示复杂的布局,这样的话,我们就得自己写布局和自定义adapter了,一般是继承于BaseAdapter,示例代码见下方.写ListView的点击事件时OnItemClickListener,onItemClick方法没有执行,导致ListView中Item条目点击事件失效,而Item中的View点击事件可以在getView方法中进行处理.导致整个Item点击失效的原因多半是由于在[你自己定义的Item中存在诸

  • Android中ListView的item点击没有反应的解决方法

    如果stu_item.xml里面包括button或者checkbox等控件,默认情况下list的item会失去焦点,导致无法响应item的事件,最常用的解决办法是在stu_item.xml的布局文件中设置descendantFocusability属性. 该属性是当一个为view获取焦点时,定义viewGroup和其子控件两者之间的关系. 属性的值有三种: beforeDescendants:viewgroup会优先其子类控件而获取到焦点 afterDescendants:viewgroup只有

随机推荐