zoukankan      html  css  js  c++  java
  • 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;
            }
    
        }

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

    源码下载:http://download.csdn.net/detail/u013293125/9734887

  • 相关阅读:
    Flutter中的剪裁
    flutter设置沉浸式状态栏
    flutter设置主题色和状态栏颜色
    Flutter更改状态栏颜色
    前端基础之JavaScript
    前端基础之CSS
    前端基础之初识 HTML
    socket及其相关(续篇)
    初识socket
    面向对象及其相关
  • 原文地址:https://www.cnblogs.com/ityizhainan/p/5976845.html
Copyright © 2011-2022 走看看