热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

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

对于ListView数据的刷新大家都知道,改变Adapter的数据源,然后调用Adapter的notifyDateSetChanged()方法即可。通过本篇文章给大家详细介绍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 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 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;
 }
 }

其实这些代码就是我上篇博文《AsyncTask实现多任务多线程下载》的例子中的,如果需要可以去下载。

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


推荐阅读
  • 理解存储器的层次结构有助于程序员优化程序性能,通过合理安排数据在不同层级的存储位置,提升CPU的数据访问速度。本文详细探讨了静态随机访问存储器(SRAM)和动态随机访问存储器(DRAM)的工作原理及其应用场景,并介绍了存储器模块中的数据存取过程及局部性原理。 ... [详细]
  • 本章将深入探讨移动 UI 设计的核心原则,帮助开发者构建简洁、高效且用户友好的界面。通过学习设计规则和用户体验优化技巧,您将能够创建出既美观又实用的移动应用。 ... [详细]
  • SQLite 动态创建多个表的需求在网络上有不少讨论,但很少有详细的解决方案。本文将介绍如何在 Qt 环境中使用 QString 类轻松实现 SQLite 表的动态创建,并提供详细的步骤和示例代码。 ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • 国内BI工具迎战国际巨头Tableau,稳步崛起
    尽管商业智能(BI)工具在中国的普及程度尚不及国际市场,但近年来,随着本土企业的持续创新和市场推广,国内主流BI工具正逐渐崭露头角。面对国际品牌如Tableau的强大竞争,国内BI工具通过不断优化产品和技术,赢得了越来越多用户的认可。 ... [详细]
  • 深入理解C++中的KMP算法:高效字符串匹配的利器
    本文详细介绍C++中实现KMP算法的方法,探讨其在字符串匹配问题上的优势。通过对比暴力匹配(BF)算法,展示KMP算法如何利用前缀表优化匹配过程,显著提升效率。 ... [详细]
  • 自学编程与计算机专业背景者的差异分析
    本文探讨了自学编程者和计算机专业毕业生在技能、知识结构及职业发展上的不同之处,结合实际案例分析两者的优势与劣势。 ... [详细]
  • MySQL 数据库迁移指南:从本地到远程及磁盘间迁移
    本文详细介绍了如何在不同场景下进行 MySQL 数据库的迁移,包括从一个硬盘迁移到另一个硬盘、从一台计算机迁移到另一台计算机,以及解决迁移过程中可能遇到的问题。 ... [详细]
  • 阅读本文大约需要3分钟。微信8.0版本的发布带来了许多令人振奋的新功能,如烟花特效和改进的悬浮窗,引发了用户的热烈反响。 ... [详细]
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
  • 作为一名程序员,从大学步入职场后,常常感受到一种难以言喻的空虚感。这种感觉或许源于对生活的不满、职业发展的瓶颈,或是日常琐事带来的压力。本文将深入探讨这种复杂的情感,并尝试寻找解决之道。 ... [详细]
  • 深入解析:阿里实战 SpringCloud 微服务架构与应用
    本文将详细介绍 SpringCloud 在微服务架构中的应用,涵盖入门、实战和案例分析。通过丰富的代码示例和实际项目经验,帮助读者全面掌握 SpringCloud 的核心技术和最佳实践。 ... [详细]
  • 本文深入探讨了C++对象模型中的一些细节问题,特别是虚拟继承和析构函数的处理。通过具体代码示例和详细分析,揭示了书中某些观点的不足之处,并提供了更合理的解释。 ... [详细]
  • 随着网络安全威胁的不断演变,电子邮件系统成为攻击者频繁利用的目标。本文详细探讨了电子邮件系统中的常见漏洞及其潜在风险,并提供了专业的防护建议。 ... [详细]
  • 微软Exchange服务器遭遇2022年版“千年虫”漏洞
    微软Exchange服务器在新年伊始遭遇了一个类似于‘千年虫’的日期处理漏洞,导致邮件传输受阻。该问题主要影响配置了FIP-FS恶意软件引擎的Exchange 2016和2019版本。 ... [详细]
author-avatar
UU常璐图_302
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有