標籤:android adapter 安卓 資料錯亂 適配器資料錯亂
在開發中經常會遇到這樣的問題,做了一個列表,給列表的每一項添加了按鈕監聽事件,但是在列表的資料很多的時候經常會出現點擊後錯亂的問題。對於這種問題,我們在程式中可能都有自己的解決辦法,但是你也許第一次發現這個問題的時候會跟我之前一樣手足無措。
那麼現在我們可以分析一下這種問題的根本原因。
首先,我們來看一下一個出錯的BaseAdapter。
package com.example.listdelectdemo;import java.util.ArrayList;import android.content.Context;import android.view.LayoutInflater;import android.view.View;import android.view.View.OnClickListener;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.Button;import android.widget.TextView;import android.widget.Toast;public class MyDataAdapter extends BaseAdapter {private Context mContext;private ArrayList<String> mStrings;private LayoutInflater mInflater;private String mStrData;public MyDataAdapter(Context c, ArrayList<String> s) {mContext = c;mStrings = s;mInflater = LayoutInflater.from(c);}@Overridepublic int getCount() {return mStrings.size();}@Overridepublic Object getItem(int position) {return mStrings.get(position);}@Overridepublic long getItemId(int position) {return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {MyViewHolder viewHolder = null;if (convertView == null) {convertView = mInflater.inflate(R.layout.item, null);viewHolder = new MyViewHolder();viewHolder.item_button_test = (Button) convertView.findViewById(R.id.item_button_test);viewHolder.item_textView_content = (TextView) convertView.findViewById(R.id.item_textView_content);convertView.setTag(viewHolder);} else {viewHolder = (MyViewHolder) convertView.getTag();}//這裡拿出來資料集合裡的當前這一項mStrData mStrData = mStrings.get(position);viewHolder.item_textView_content.setText(mStrData);viewHolder.item_button_test.setText("點擊");//這裡給item的button設定了點擊監聽事件viewHolder.item_button_test.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) { //這裡toast出來的mStrData卻不是點擊的那一項Toast.makeText(mContext, "您點擊了-" + mStrData, Toast.LENGTH_LONG).show();}});return convertView;}class MyViewHolder {TextView item_textView_content;Button item_button_test;}}
然後,我們分析一下原因,相信老程式員都可以看出問題的所在:
mStrData = mStrings.get(position);
getView方法第一次被調用的時候,將集合中的當前項資料拿出來付給了成員變數mStrData,程式繼續往下執行:
viewHolder.item_button_test.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) { //這裡toast出來的mStrData卻不是點擊的那一項Toast.makeText(mContext, "您點擊了-" + mStrData, Toast.LENGTH_LONG).show();}});
這裡給item的按鈕添加了監聽事件,但是要注意:程式並不會回調監聽事件中的
@Overridepublic void onClick(View v) { //這裡toast出來的mStrData卻不是點擊的那一項 Toast.makeText(mContext, "您點擊了-" + mStrData, Toast.LENGTH_LONG).show();}
而是會繼續回調getView方法。
等到列表即將被載入完成,也就是最後一次回調getView方法時,成員變數mStrData會被最後一次賦值,
那麼,getView方法每回調一次,mStrData的值就會被重新賦一次。
然後,當我們點擊按鈕,就會回調監聽的onClick方法,這時候執行toast:
Toast.makeText(mContext, "您點擊了-" + mStrData, Toast.LENGTH_LONG).show();
此時的mStrData就是只能是最後一次賦的值了,出錯就是必然的。
那麼,來看一下我的解決方案:
package com.example.listdelectdemo;import java.util.ArrayList;import android.content.Context;import android.view.LayoutInflater;import android.view.View;import android.view.View.OnClickListener;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.Button;import android.widget.TextView;import android.widget.Toast;public class MyDataAdapter extends BaseAdapter {private Context mContext;private ArrayList<String> mStrings;private LayoutInflater mInflater;private String mStrData;public MyDataAdapter(Context c, ArrayList<String> s) {mContext = c;mStrings = s;mInflater = LayoutInflater.from(c);}@Overridepublic int getCount() {return mStrings.size();}@Overridepublic Object getItem(int position) {return mStrings.get(position);}@Overridepublic long getItemId(int position) {return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {MyViewHolder viewHolder = null;if (convertView == null) {convertView = mInflater.inflate(R.layout.item, null);viewHolder = new MyViewHolder();viewHolder.item_button_test = (Button) convertView.findViewById(R.id.item_button_test);viewHolder.item_textView_content = (TextView) convertView.findViewById(R.id.item_textView_content);convertView.setTag(viewHolder);} else {viewHolder = (MyViewHolder) convertView.getTag();}mStrData = mStrings.get(position);viewHolder.item_textView_content.setText(mStrData);viewHolder.item_button_test.setText("點擊");// 給item的按鈕設定點擊監聽,建立一個監聽的實作類別,並傳入當前的positionviewHolder.item_button_test.setOnClickListener(new MyAdapterListener(position));return convertView;}class MyViewHolder {TextView item_textView_content;Button item_button_test;}class MyAdapterListener implements OnClickListener {private int position;public MyAdapterListener(int pos) {position = pos;}@Overridepublic void onClick(View v) {Toast.makeText(mContext, "您點擊了-" + mStrings.get(position), Toast.LENGTH_LONG).show();}}}
這時候就不會出現問題。不同的地方就在於給item的button添加點擊事件的時候是每次都建立一個新的MyAdapterListener對象來實現這個監聽。那麼在new這個MyAdapterListener對象的時候給他傳入一個position做為標記,這樣每一個item都會有一個屬於自己的監聽類,我們就可以在這個監聽類中做一些自己的邏輯處理,就不會出現錯亂的問題。
這個方案只能作為一個參考方案,有一個弊端就是在列表資料多的時候,會建立很多新的對象,而佔用記憶體。那麼大家有什麼更好的方案可以分享分享。
本文出自 “小明的碼路” 部落格,轉載請與作者聯絡!
Android中ListView的item按鈕監聽錯亂問題解決辦法