你的位置:首页 > 操作系统

[操作系统]Android——Listview不用notifydatasetchanged更新数据的方法


一、介绍 
先来介绍一下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.

<LinearLayout  android:orientation="vertical"  ="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" ="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.

<??><LinearLayout ="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;    }  }

只有这两个地方改变了一下。