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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...