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

    }

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

时间: 2024-10-08 15:34:52

Android——Listview不用notifydatasetchanged更新数据的方法的相关文章

Android SDK无法显示更新列表解决方法

解决办法: 第一步: 打开Windows中C:\WINDOWS\system32\drivers\etc\hosts,然后添加以下内容: 203.208.46.146 dl.google.com 203.208.46.146 dl-ssl.google.com 74.125.113.121 developer.android.com 第二步: 打开Android SDK Manager,选上方的菜单Tools,进入Options,在"Force https://- "前面打钩,就可以更

delphi xe6 android ListView增加 Header或Footer 的方法

var  Item1: TListViewItem;begin    Item1 := ListView1.Items.Add;    Item1.Purpose:=TListItemPurpose.Header;//    Item1.Purpose:=TListItemPurpose.Footer;    Item1.text:=' xxxx';//Header或Footer显示的内容end; delphi xe6 android ListView增加 Header或Footer 的方法,布

android子线程中更新UI的方法

在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法: 参考:Android子线程 方法一:用Handler 1.主线程中定义Handler: Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) { super.handleMessage(msg); switch (msg.what) { case 0: //

ThinkPHP 更新数据 save方法

ThinkPHP save() 方法 ThinkPHP 中使用 save() 方法来更新数据库,并且也支持连贯操作的使用. 例子: public function update(){ header("Content-Type:text/html; charset=utf-8"); $Dao = M("User"); // 需要更新的数据 $data['email'] = '[email protected]'; // 更新的条件 $condition['userna

AJAX制作JSON格式的实时更新数据的方法

之前有写过这样的文章,但是出现了几个问题,第一,如果每秒都像数据库发送请求势必会造成服务器的压力过大,第二,如果使用JS的话,是不可以取得系统时间的,因为JS运行在客户端,所以只能取得客户端时间, 如此的话,就需要在后台计算好时间差,然后前台得到时间差进行倒计时计算,并且更新数据 JS代码如下: var i = 0; var ballNum = ""; var flag = true; var timer_showCountDown = null; var leftTime; func

关于ListView中notifyDataSetChanged()刷新数据不更新原因

使用Listview的时候: 当要动态显示更改后的数据(例如数据库改动), 很多人应该都用过notifyDataSetChanged();这个方法来刷新Listview,显示改后的数据. 这时候就要注意了: ...... private ArrayList<HashMap<String, Object>> usersList; ...... usersList= query(......);     //根据查询函数获取一个ArrayList并赋值给绑定到Adapter的数据源us

ListView中notifyDataSetChanged()刷新数据不更新原因分析

情景分析: 在开发过程中,我们经常会使用notifyDataSetChanged()实时的让ListView进行刷新操作,但是有的时候却会发现ListView不再刷新了,有的时候却又可以自动刷新. 原因分析: 如上图所示,传递给Adapter的List引用一开始为引用a,但是,当我们在程序中执行刷新更换数据的时候,如果一不小心将List的引用换成了引用b的话,就会导致上述情景的发生.那是因为Adapter刷新的时候,还是用得引用a的数据,而不是引用b: 解决思路: 思路1: 在Adapter中为

【转】关于ListView中notifyDataSetChanged()刷新数据不更新原因

使用Listview的时候: 当要动态显示更改后的数据(例如数据库改动), 很多人应该都用过notifyDataSetChanged();这个方法来刷新Listview,显示改后的数据. 这时候就要注意了:                  ......                  private ArrayList<HashMap<String, Object>> usersList;                  ......                  use

Android ListView的适配器 Adapter 中GetView方法调用次数大于数据条数的问题

情况描述: 今天在写数据展示的时候,用了ListView,在它的适配器当中,发现getview方法执行的次数是数据条目的N倍(我这边显示的是4倍,这个倍数值不知道会不会变化),这显然是不科学的! 查阅资料,发现是ListView的布局问题.我把 android:layout_height="wrap_content"高度设置成了wrap_content.而wrap_content值使得ListView没有取到实际的高度,他还要根据计算才能确定,而每一次计算应该会触发listview的渲