Android——Listview不用notifydatasetchanged更新數據的方法

来源:http://www.cnblogs.com/ityizhainan/archive/2016/10/19/5976845.html
-Advertisement-
Play Games

一、介紹 先來介紹一下listview更新數據的幾種方法,目前我知道的方法有如下幾種: 1. 每次更新數據時都調用listview.setadapter(); 2. 每次更新數據時都調用adapter.notifydatasetchanged(); 3. 在自定義的adapter里添加更新函數upd ...


一、介紹 
先來介紹一下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;
        }

    }

只有這兩個地方改變了一下。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • AToolActivity.java SmsBackUp.java 1 package com.itheima.mobilesafe74.engine; 2 3 import java.io.File; 4 import java.io.FileOutputStream; 5 import java ...
  • 新接觸一門程式或者開發框架,我一般都要先弄清楚程式的入口在哪裡,程式怎麼運行的;建立一個項目後,各個文件夾有什麼作用以及如何使用等等。理清楚這些東西對以後開發是很有好處的,古話說得好,工欲善其事,必先利其器。本文主要總結android程式的入口點和程式的運行流程(初識),android項目中各文件夾 ...
  • ReactiveX是Reactive Extensions的縮寫,一般簡寫為Rx,最初是LINQ的一個擴展,由微軟的架構師Erik Meijer領導的團隊開發,在2012年11月開源,Rx是一個編程模型,目標是提供一致的編程介面,幫助開發者更方便的處理非同步數據流,Rx庫支持.NET、JavaScri ...
  • 前言 Swift已推出數年,與Objective-C相比Swift的語言機制及使用簡易程度上更接地氣,大大降低了iOS入門門檻。當然這對新入行的童鞋沒來講,的確算是福音,但對於整個iOS編程從業者來講,真真是,曾幾何時“高大上”,轉瞬之間“矮矬窮”。再加上培訓班橫行,批量批發之下,iOS再也看不到當 ...
  • 移動應用在處理網路資源時,一般都會做離線緩存處理,其中以圖片緩存最為典型,其中很流行的離線緩存框架為SDWebImage。 但是,離線緩存會占用手機存儲空間,所以緩存清理功能基本成為資訊、購物、閱讀類app的標配功能。 今天介紹的離線緩存功能的實現,主要分為緩存文件大小的獲取、清除緩存文件的實現。 ...
  • 一、字元串轉JSON 在網路請求時,如果服務端返回的是字元串,那麼就需要我們自己封裝一個類,將請求下來的字元串轉換成json對象,從而存入模型中。 註意: 字元串中如果含有一些特殊轉意符(如\n、\t等),需要先對字元串進行處理。 示例代碼如下: 二、圖片拉伸 三、Label文字自適應frame 方 ...
  • GCD定時器 圖片上繪製文字 查找一個視圖的所有子視圖 計算文件大小 UIView的設置部分圓角 計算字元串字元長度,一個漢字算兩個字元 防止滾動視圖手勢覆蓋側滑手勢 去掉導航欄返回的標題 字元串中是否含有中文 dispatch_group的使用 UITextField每四位加一個空格,實現代理 獲 ...
  • 軟引用 和 弱引用 1. SoftReference<T>:軟引用-->當虛擬機記憶體不足時,將會回收它指向的對象;需要獲取對象時,可以調用get方法。 2. WeakReference<T>:弱引用-->隨時可能會被垃圾回收器回收,不一定要等到虛擬機記憶體不足時才強制回收。要獲取對象時,同樣可以調用g ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...