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 0; } @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(0); // 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 0; } @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(0); // cache the view return convertView; } public static class ViewHolder { ProgressBar pb; Button btn; } } </model></model> Activity ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 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 = 0;i<100;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 = 0;i<=100;i++){ updateProgressInUiThread(model, i,position); try { Thread.sleep(50); } 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
效果為:

