首页 > 代码库 > Android——Listview不用notifydatasetchanged更新数据的方法

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.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;
        }

    }

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

Android——Listview不用notifydatasetchanged更新数据的方法