Android——Listview不用notifydatasetchanged更新資料的方法,androidlistview更新
一、介紹
先來介紹一下listview更新資料的幾種方法,目前我知道的方法有如下幾種:
1. 每次更新資料時都調用listview.setadapter();
2. 每次更新資料時都調用adapter.notifydatasetchanged();
3. 在自訂的adapter裡添加更新函數update;
部落格撰寫人:It一zhai男
轉載請標明地址:http://blog.csdn.net/u013293125/article/details/52858396
這裡,我們將會一個一個來介紹,順便說一句,對ListView的工作原理和機制不明白的可以看看這篇文章:http://blog.csdn.net/guolin_blog/article/details/44996879(大神都是看原碼的,在此獻上我的膝蓋)。
1. 每次更新資料時都調用listview.setadapter();
這個方法是效率最低的,因為它不管你其它的資料需不需要重新整理,它都會將所有的資料重新整理一遍,也就是說將整個listview重新整理一遍,估計會一點Android的人都不會用這種方法,但我們還是將其列出來,可以與其它方法進行對比。
1.1 先上:
點擊更新後:
1.2 activity_main.xml檔案
<LinearLayout android:orientation="vertical" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/container" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.listviewupdate.MainActivity" tools:ignore="MergeRootFrame" xmlns:android="http://schemas.android.com/apk/res/android"> <Button android:id="@+id/btn" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="更新"/> <ListView android:id="@+id/listview" android:layout_width="match_parent" android:layout_height="wrap_content"> </ListView></LinearLayout>
1.3 item.xml檔案:
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal" > <TextView android:id="@+id/tv1" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <TextView android:id="@+id/tv2" android:layout_width="wrap_content" android:layout_height="wrap_content" /></LinearLayout>
1.4 MainActivity.java檔案:
package com.example.listviewupdate;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import java.util.Map;import android.support.v7.app.ActionBarActivity;import android.support.v7.app.ActionBar;import android.support.v4.app.Fragment;import android.app.Activity;import android.content.Context;import android.os.Bundle;import android.view.LayoutInflater;import android.view.Menu;import android.view.MenuItem;import android.view.View;import android.view.View.OnClickListener;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.Button;import android.widget.ListView;import android.widget.TextView;import android.os.Build;public class MainActivity extends Activity { private ListView listview; private List<Map<String, Object>>list = new ArrayList<Map<String,Object>>(); private MyAdapter adapter; private Button btn; Map<String, Object>map; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); listview = (ListView) findViewById(R.id.listview); //初始化資料 for (int i = 0; i < 8; i++) { map = new HashMap<String, Object>(); map.put("Id", "100"+i); map.put("Name","Name_"+i); list.add(map); } adapter = new MyAdapter(this, list); listview.setAdapter(adapter); btn = (Button) findViewById(R.id.btn); //比如說,要更新listview裡第三行的Name,但下面的做法是重新載入了一下adapter //也就是說它重新整理了整個listview,不管其他的資料需不需要更新; btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub map = list.get(2); map.put("Name", "更新的名字"); //這裡MyAdapter的第一個參數不用this原因是因為這裡是一個匿名內部類, //this指向的是onClick裡 adapter = new MyAdapter(MainActivity.this, list); listview.setAdapter(adapter); } }); } //自訂adapter public class MyAdapter extends BaseAdapter{ List<Map<String, Object>>list; LayoutInflater inflater; public MyAdapter(Context context,List<Map<String, Object>>list){ this.list = list; inflater = LayoutInflater.from(context); } @Override public int getCount() { // TODO Auto-generated method stub return list.size(); } @Override public Object getItem(int position) { // TODO Auto-generated method stub return list.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 ViewHolder viewHolder; if(convertView==null){ convertView = inflater.inflate(R.layout.item, null); viewHolder = new ViewHolder(); viewHolder.tv1 = (TextView) convertView.findViewById(R.id.tv1); viewHolder.tv2 =(TextView) convertView.findViewById(R.id.tv2); convertView.setTag(viewHolder); }else{ viewHolder = (ViewHolder) convertView.getTag(); } viewHolder.tv1.setText(list.get(position).get("Id").toString()); viewHolder.tv2.setText(list.get(position).get("Name").toString()); return convertView; } } //輔助類 class ViewHolder{ TextView tv1; TextView tv2; }}
2. 每次更新資料時都調用adapter.notifydatasetchanged();
如果適配器的內容改變,notifyDataSetChanged方法將會通過一個外部方法強制調用getView來重新整理每個Item的內容。(這句話是網上看到的,說的也不是太清楚,看了一下notifydatasetchanged()源碼也不是很清楚。),這個方法在資料量比較少,重新整理頻率比較慢的情況下還是不錯的。
布局什麼的都和上面一樣,這裡就只發MainActivity.java裡面的內容。
btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub map = list.get(2); map.put("Name", "更新的名字"); //只用這裡改變了 adapter.notifyDataSetChanged(); } }); }
3. 在自訂的adapter裡添加更新函數update;
這種方法會更新你指定地方指定位置的資料,比如說Listview的第三個item項的第二個TextView,那麼它就只更新這裡,其他的不會更新(通過網上資料和個人理解)。layout布局都是一樣的,這裡主要是自訂adapter裡的改變。
btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub map = list.get(2); map.put("Name", "更新的名字"); //只有這裡改變 adapter.update(2, listview); } });
public class MyAdapter extends BaseAdapter{ List<Map<String, Object>>list; LayoutInflater inflater; public MyAdapter(Context context,List<Map<String, Object>>list){ this.list = list; inflater = LayoutInflater.from(context); } @Override public int getCount() { // TODO Auto-generated method stub return list.size(); } @Override public Object getItem(int position) { // TODO Auto-generated method stub return list.get(position); } @Override public long getItemId(int position) { // TODO Auto-generated method stub return position; } public void update(int index,ListView listview){ //得到第一個可見item項的位置 int visiblePosition = listview.getFirstVisiblePosition(); //得到指定位置的視圖,對listview的緩衝機制不清楚的可以去瞭解下 View view = listview.getChildAt(index - visiblePosition); ViewHolder holder = (ViewHolder) view.getTag(); holder.tv2 = (TextView) view.findViewById(R.id.tv2); setData(holder,index); } private void setData(ViewHolder holder,int index){ Map<String, Object>map = list.get(index); holder.tv2.setText(map.get("Name").toString()); } @Override public View getView(int position, View convertView, ViewGroup parent) { // TODO Auto-generated method stub ViewHolder viewHolder; if(convertView==null){ convertView = inflater.inflate(R.layout.item, null); viewHolder = new ViewHolder(); viewHolder.tv1 = (TextView) convertView.findViewById(R.id.tv1); viewHolder.tv2 =(TextView) convertView.findViewById(R.id.tv2); convertView.setTag(viewHolder); }else{ viewHolder = (ViewHolder) convertView.getTag(); } viewHolder.tv1.setText(list.get(position).get("Id").toString()); viewHolder.tv2.setText(list.get(position).get("Name").toString()); return convertView; } }
只有這兩個地方改變了一下。