1.简介

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

->ListView局部刷新问题的引入:

假设我们的ListView的Item中有一个进度条(ProgressBar)和一个按钮,当我们点击一下按钮,进度条就会从0到100进行刷新,而且一般都需要在1s以内完成刷新过程,也就是说:在ListView的任一个Item中,触发了Button的事件之后,在1s以内,或者更短的时间,ProgressBar需要刷新100次。显然,如果我们使用修改数据源,调用notifyDataSetChanged()进行刷新的机制明显是不恰当的,效率极低,而且不一定凑效。那么,我们自然想到当点击的时候,希望能够获取到点击后的View内部的ProgressBar控件的对象,然后直接调用progressBar的setProgress就可以了,本以为这样就大功告成了。忽然,你会发现,当progressBar正在更新的时候,此时,往下滑listview,突然发现下面的某个进度条也在更新。仔细一分析,还真有道理,因为ListView中的View是复用的,当你向下滑动listview的时候,你此时操作的progressBar对象,已经不是刚才点击的那个Item了,因为很多Item复用一个View。那么如何解决这个问题呢?

2.解决方案

记录点击的Item的position,然后在更新过程中,不断的判断,该position是不是介于可见的Item之间,如果是,则更新,否者,不更新。

private void updateProgressPartly(int progress,int position){
int firstVisiblePosition = listview.getFirstVisiblePosition();
int lastVisiblePosition = listview.getLastVisiblePosition();
if(position>=firstVisiblePosition && position<=lastVisiblePosition){
View view = listview.getChildAt(position - firstVisiblePosition);
if(view.getTag() instanceof ViewHolder){
ViewHolder vh = (ViewHolder)view.getTag();
vh.pb.setProgress(progress);
}
}
}

其他相关代码:

ListAdapter

/*
* $filename: ListAdapter.java,v $
* $Date: 2014-9-19 $
* Copyright (C) ZhengHaibo, Inc. All rights reserved.
* This software is Made by Zhenghaibo.
*/
package net.mobctrl.listviewdemo; import java.util.ArrayList;
import java.util.List; import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ProgressBar; /*
*@author: ZhengHaibo
*blog: http://blog.csdn.net/nuptboyzhb
*mail: zhb931706659@126.com
*web: http://www.mobctrl.net
*2014-9-19 Nanjing,njupt,China
*/
public class ListAdapter extends BaseAdapter { private List<model> datas;
private Context context; private UpdateCallback updateCallback; public UpdateCallback getUpdateCallback() {
return updateCallback;
} public void setUpdateCallback(UpdateCallback updateCallback) {
this.updateCallback = updateCallback;
} public ListAdapter(Context context) {
datas = new ArrayList<model>();
this.context = context;
} public void addData(Model model) {
datas.add(model);
notifyDataSetChanged();
} @Override
public int getCount() {
// TODO Auto-generated method stub
return datas.size();
} @Override
public Object getItem(int pos) {
// TODO Auto-generated method stub
return datas.get(pos);
} @Override
public long getItemId(int arg0) {
// TODO Auto-generated method stub
return ;
} @Override
public View getView(final int pos, View convertView, ViewGroup viewGroup) {
final Model model = datas.get(pos);
ViewHolder viewHolder = null;
if (null == convertView) {
viewHolder = new ViewHolder();
convertView = LayoutInflater.from(context).inflate(
R.layout.list_item_layout, null);
viewHolder.pb = (ProgressBar) convertView
.findViewById(R.id.pb_show);
viewHolder.btn = (Button) convertView.findViewById(R.id.btn);
convertView.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) convertView.getTag();
convertView.setTag(viewHolder);
}
viewHolder.btn.setText(model.getName());
viewHolder.btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if(null != updateCallback){
updateCallback.startProgress(model,pos);
}
}
});
viewHolder.pb.setProgress();
// cache the view
return convertView;
} public static class ViewHolder {
ProgressBar pb;
Button btn;
} }

Activity

/*
* $filename: ListAdapter.java,v $
* $Date: 2014-9-19 $
* Copyright (C) ZhengHaibo, Inc. All rights reserved.
* This software is Made by Zhenghaibo.
*/
package net.mobctrl.listviewdemo; import java.util.ArrayList;
import java.util.List; import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ProgressBar; /*
*@author: ZhengHaibo
*blog: http://blog.csdn.net/nuptboyzhb
*mail: zhb931706659@126.com
*web: http://www.mobctrl.net
*2014-9-19 Nanjing,njupt,China
*/
public class ListAdapter extends BaseAdapter { private List<model> datas;
private Context context; private UpdateCallback updateCallback; public UpdateCallback getUpdateCallback() {
return updateCallback;
} public void setUpdateCallback(UpdateCallback updateCallback) {
this.updateCallback = updateCallback;
} public ListAdapter(Context context) {
datas = new ArrayList<model>();
this.context = context;
} public void addData(Model model) {
datas.add(model);
notifyDataSetChanged();
} @Override
public int getCount() {
// TODO Auto-generated method stub
return datas.size();
} @Override
public Object getItem(int pos) {
// TODO Auto-generated method stub
return datas.get(pos);
} @Override
public long getItemId(int arg0) {
// TODO Auto-generated method stub
return ;
} @Override
public View getView(final int pos, View convertView, ViewGroup viewGroup) {
final Model model = datas.get(pos);
ViewHolder viewHolder = null;
if (null == convertView) {
viewHolder = new ViewHolder();
convertView = LayoutInflater.from(context).inflate(
R.layout.list_item_layout, null);
viewHolder.pb = (ProgressBar) convertView
.findViewById(R.id.pb_show);
viewHolder.btn = (Button) convertView.findViewById(R.id.btn);
convertView.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) convertView.getTag();
convertView.setTag(viewHolder);
}
viewHolder.btn.setText(model.getName());
viewHolder.btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if(null != updateCallback){
updateCallback.startProgress(model,pos);
}
}
});
viewHolder.pb.setProgress();
// cache the view
return convertView;
} public static class ViewHolder {
ProgressBar pb;
Button btn;
} }
</model></model> Activity ? package net.mobctrl.listviewdemo; import net.mobctrl.listviewdemo.ListAdapter.ViewHolder; import org.androidannotations.annotations.AfterViews;
import org.androidannotations.annotations.EActivity;
import org.androidannotations.annotations.UiThread;
import org.androidannotations.annotations.ViewById; import android.app.Activity;
import android.view.View;
import android.widget.ListView;
/**
* @author 郑海波
* @webset http://www.mobctrl.net
* ListView的局部刷新
*/
@EActivity(R.layout.activity_main)
public class MainActivity extends Activity implements UpdateCallback{ @ViewById
ListView listview; private ListAdapter adapter; @AfterViews
void afterViews() {
adapter = new ListAdapter(this);
adapter.setUpdateCallback(this);
listview.setAdapter(adapter);
initDatas();
} private void initDatas() {
for(int i = ;i<;i++){
Model model = new Model(i, <click> --> );
adapter.addData(model);
}
} @Override
public void startProgress(final Model model,final int position) {
/** start the Thread to update the Progress*/
new Thread(new Runnable() { @Override
public void run() {
for(int i = ;i<=;i++){
updateProgressInUiThread(model, i,position);
try {
Thread.sleep();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}).start();
} @UiThread
void updateProgressInUiThread(Model model,int progress,int position){
updateProgressPartly(progress,position);
} private void updateProgressPartly(int progress,int position){
int firstVisiblePosition = listview.getFirstVisiblePosition();
int lastVisiblePosition = listview.getLastVisiblePosition();
if(position>=firstVisiblePosition && position<=lastVisiblePosition){
View view = listview.getChildAt(position - firstVisiblePosition);
if(view.getTag() instanceof ViewHolder){
ViewHolder vh = (ViewHolder)view.getTag();
vh.pb.setProgress(progress);
}
}
} }

布局:

整个项目的源代码:https://github.com/nuptboyzhb/ListViewPartRefreash

效果为:

android:ListView的局部刷新的更多相关文章

  1. ListView的局部刷新

    有的列表可能notifyDataSetChanged()代价有点高,最好能局部刷新. 局部刷新的重点是,找到要更新的那项的View,然后再根据业务逻辑更新数据即可. private void upda ...

  2. Android ListView下拉刷新时卡的问题解决小技巧

    问题:ListView下拉刷新时看上去非常的卡 解决方案: 在BaseAdapter的getView方法中,有三个参数 public View getView(int position, View c ...

  3. Android listview下拉刷新 SwipeRefreshLayout

    今天在Google+上看到了SwipeRefreshLayout这个名词,遂搜索了下,发现竟然是刚刚google更新sdk新增加的一个widget,于是赶紧抢先体验学习下. SwipeRefreshL ...

  4. android ListView下拉刷新 上拉加载更多

    背景 最近在公司的项目中要使用到ListView的下拉刷新和上拉加载更多(貌似现在是个项目就有这个功能!哈哈),其实这个东西GitHub上很多,但是我感觉那些框架太大,而且我这个项目只用到了ListV ...

  5. Android listview局部刷新和模拟应用下载(zhu)

    在android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用notifyDataSetChanged()这个函数,但是它会更新listview中所有可 ...

  6. 转:android listview局部刷新和模拟应用下载

    在 android开发中,listview是比较常用的一个组件,在listview的数据需要更新的时候,一般会用 notifyDataSetChanged()这个函数,但是它会更新listview中所 ...

  7. Android 结合实际项目学会ListView局部刷新和相关知识《一》

    转载本专栏博客,请注明出处:道龙的博客 最近在公司参与的项目中有一个界面需要做局部UI更新处理,把其化烦为简为Demoi形式写在这里.我们还是运行该Demo,知道ListView局部刷新的使用场景:( ...

  8. RecyclerView局部刷新那点事

    1.局部刷新的引入 提到RecyclerView,我们首先想到的是ListView,对于ListView的局部刷新,我们之前已经有解决方案,[android:ListView的局部刷新]当时的解决方案 ...

  9. Android ListView 自动加载更多

    Android ListView下拉刷新 ListView是我们经常用来展示数据的一个控件,但是由于我们手机的性能和流量的问题,往往我们从服务器中取数据,不能一次性将数据取出来,比如一个新闻的手机AP ...

随机推荐

  1. .net自定义控件

    一.[.net自定义控件]TextBox控件重写 之NumTextBox 参考博客:http://www.yongfa365.com/Item/NumTextBox.html 二.[.net自定义控件 ...

  2. PNG文件格式具体解释

      PNG文件结构分析(上:了解PNG文件存储格式) 前言 我们都知道,在进行J2ME的手机应用程序开发的时候,在图片的使用上,我们能够使用PNG格式的图片(甚至于在有的手机上,我们仅仅能够使用PNG ...

  3. 打造支持apk下载和html5缓存的 IIS(配合一个超简单的android APP使用)具体解释

    为什么要做这个看起来不靠谱的东西呢? 由于刚学android开发,还不能非常好的熟练控制android界面的编辑和操作,所以我的一个急着要的运用就改为html5版本号了,反正这个运用也是须要从serv ...

  4. HTTP in iOS你看我就够

    HTTP属于老话题了,在项目中我们经常需要往服务端发POST或者GET请求,但是对于HTTP的了解不应只局限于此.千里之行,始于足下.越想走的远,基本原理就应该了解的透彻全面一些,仅仅停留在使用ASI ...

  5. Facade 门面模式 外观模式

    简介 作用: (1)封装一组交互类,一致地对外提供接口 (2)封装子系统,简化子系统调用 JDK中体现:java.util.logging包 java.lang.Class javax.faces.w ...

  6. VS2008 快捷键大全--------<<转>>

    VS2008 快捷键大全   Ctrl+E,D ----             格式化全部代码 Ctrl+K,F ----             格式化选中的代码 CTRL + SHIFT + B ...

  7. js的new操作符

    1.创建一个空对象,并且 this 变量引用该对象,同时还继承了该函数的原型. 2.属性和方法被加入到 this 引用的对象中. 3.新创建的对象由 this 所引用,并且最后隐式的返回 this . ...

  8. HTML简要内容

    1.  html基础 html是用来制作网页的标记语言,不需编译,直接由浏览器执行.大小写不敏感,推荐使用小写.html文件必须使用html或htm为文件名后缀. html主体结构: (1)DTD头: ...

  9. 高健壮性css---Float详细

    (一)关于float 首先我们了解到,CSS网页布局的原理,就是按照HTML代码中对象声明的顺序,以流布局的方式来显示它,而流布局就不得不说到float浮动技术..在HTML中的所有对象,默认分为两种 ...

  10. Java枚举类型理解

    Enum格式理解 Enum的格式可以看做跟class关键字一样 class的定义格式如下: public class abc{ } enum的定义格式如下: Public enum abc { } 引 ...