衆所周知Listview和Gridview的刷新界面的方式是調用adapter.notifyDataSetChanged()進行界面刷新。
但是此方法有其弊端,他是將界面中的數據全部刷新一遍,不論數據有沒有變化。如果listview加載了很多的數據(如:100條)
在進行刷新時就會造成很大的系統開銷如何像qq空間個人動態那樣點贊只刷新一條呢:
主要原理:
對listview的某一個item進行刷新
1.要獲取要刷新的item當前索引position和數據
2.對獲取的數據進行重置
3.將重置的數據放到adapter中的數據集的原來的位置(根據position刷新原數據集的某一條數據)
4.在listview中獲取需要刷新的子item的view
5.從更新過的數據集中獲取新數據,更新viwe中的數據(handler中操作,實現界面的刷新)
功能如下,代碼中有詳細註釋:
public class MainActivity extends Activity
{
private ArrayList<MyListItem> list = null;
private ListView lv;
private MyListAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intitData();
lv = (ListView) findViewById(R.id.listView1);
adapter = new MyListAdapter(list, getApplicationContext());
adapter.setListView(lv);
lv.setAdapter(adapter);
lv.setOnItemClickListener(new OnItemClickListener()
{
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id)
{
// 獲取listview中點擊item的數據
MyListItem item = (MyListItem) parent.getItemAtPosition(position);
Log.i("eee", item.getData() + " == " + item.getPosition());
// 更新數據
item.setData("update item " + position);
// 更新界面
adapter.updateItemData(item);
}
});
}
/**
* 初始化數據
*/
private void intitData()
{
list = new ArrayList<MyListItem>();
for (int i = 0; i < 20; i++)
{
MyListItem item = new MyListItem();
item.setData("item " + i);
item.setPosition(i);
list.add(item);
}
}
/**
* 自定義item數據類型
*/
class MyListItem
{
/**
* 數據id
*/
private int dataId;
/**
* 數據
*/
private String data;
public int getPosition()
{
return dataId;
}
public void setPosition(int position)
{
this.dataId = position;
}
public String getData()
{
return data;
}
public void setData(String data)
{
this.data = data;
}
}
}
activity進行調用,功能操作主要封裝在adapter中如下:
public class MyListAdapter extends BaseAdapter
{
/**
* listview中的數據集
*/
private ArrayList<MyListItem> mDataList;
private Context mContext;
private ListView mListView;
public MyListAdapter(ArrayList<MyListItem> list, Context cont)
{
this.mDataList = list;
this.mContext = cont;
}
/**
* 設置listview對象
*
* @param lisv
*/
public void setListView(ListView lisv)
{
this.mListView = lisv;
}
/**
* update listview 單條數據
*
* @param item 新數據對象
*/
public void updateItemData(MyListItem item)
{
Message msg = Message.obtain();
int ids = -1;
// 進行數據對比獲取對應數據在list中的位置
for (int i = 0; i < mDataList.size(); i++)
{
if (mDataList.get(i).getPosition() == item.getPosition())
{
ids = i;
}
}
msg.arg1 = ids;
// 更新mDataList對應位置的數據
mDataList.set(ids, item);
// handle刷新界面
han.sendMessage(msg);
}
@SuppressLint("HandlerLeak")
private Handler han = new Handler()
{
public void handleMessage(android.os.Message msg)
{
updateItem(msg.arg1);
};
};
/**
* 刷新指定item
*
* @param index item在listview中的位置
*/
private void updateItem(int index)
{
if (mListView == null)
{
return;
}
// 獲取當前可以看到的item位置
int visiblePosition = mListView.getFirstVisiblePosition();
// 如添加headerview後 firstview就是hearderview
// 所有索引+1 取第一個view
// View view = listview.getChildAt(index - visiblePosition + 1);
// 獲取點擊的view
View view = mListView.getChildAt(index - visiblePosition);
TextView txt = (TextView) view.findViewById(R.id.textView1);
// 獲取mDataList.set(ids, item);更新的數據
MyListItem data = (MyListItem) getItem(index);
// 重新設置界面顯示數據
txt.setText(data.getData());
}
@Override
public int getCount()
{
// TODO Auto-generated method stub
return mDataList.size();
}
@Override
public Object getItem(int position)
{
// TODO Auto-generated method stub
return mDataList.get(position);
}
@Override
public long getItemId(int position)
{
// TODO Auto-generated method stub
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent)
{
// TODO Auto-generated method stub
if (convertView == null)
{
convertView = LayoutInflater.from(mContext).inflate(R.layout.list_item, null);
}
TextView txt = (TextView) convertView.findViewById(R.id.textView1);
txt.setText(mDataList.get(position).getData());
return convertView;
}
}
由於listview與gridview功能相似,只是顯示方式不同,原理一樣,需要的同學可以自己修改一下試試
源碼下載