編輯:關於android開發
一、介紹
先來介紹一下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開發之自定義控件(二)---onLayout詳解 話說一個乞丐在看一個程序員寫程序,程序員遇到一個問題怎麼都解決不了,這時乞丐說
(轉)android底部彈出iOS7風格對話選項框(QQ對話框)--第三方開源--IOS_Dialog_Library,android刷ios7本文轉載於:http://
懸浮對話框的實現,懸浮對話框實現直接上代碼: private void setDialog(){ View view = getLayoutIn
Android屬性動畫 屬性動畫系統是一個健壯 的框架,它幾乎可以允許把任何對象變成動畫。可以根據時間的推移來改變任何對象的屬性來定義一個動畫,而不用關心該對象是否要繪制