您好,登錄后才能下訂單哦!
在開發中經常會遇到這樣的問題,做了一個列表,給列表的每一項添加了按鈕監聽事件,但是在列表的數據很多的時候經常會出現點擊后錯亂的問題。對于這種問題,我們在程序中可能都有自己的解決辦法,但是你也許第一次發現這個問題的時候會跟我之前一樣手足無措。
那么現在我們可以分析一下這種問題的根本原因。
首先,我們來看一下一個出錯的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); } @Override public int getCount() { return mStrings.size(); } @Override public Object getItem(int position) { return mStrings.get(position); } @Override public long getItemId(int position) { return position; } @Override public 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() { @Override public 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() { @Override public void onClick(View v) { //這里toast出來的mStrData卻不是點擊的那一項 Toast.makeText(mContext, "您點擊了-" + mStrData, Toast.LENGTH_LONG).show(); } });
這里給item的按鈕添加了監聽事件,但是要注意:程序并不會回調監聽事件中的
@Override public 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); } @Override public int getCount() { return mStrings.size(); } @Override public Object getItem(int position) { return mStrings.get(position); } @Override public long getItemId(int position) { return position; } @Override public 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的按鈕設置點擊監聽,創建一個監聽的實現類,并傳入當前的position viewHolder.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; } @Override public void onClick(View v) { Toast.makeText(mContext, "您點擊了-" + mStrings.get(position), Toast.LENGTH_LONG).show(); } } }
這時候就不會出現問題。不同的地方就在于給item的button添加點擊事件的時候是每次都創建一個新的MyAdapterListener對象來實現這個監聽。那么在new這個MyAdapterListener對象的時候給他傳入一個position做為標記,這樣每一個item都會有一個屬于自己的監聽類,我們就可以在這個監聽類中做一些自己的邏輯處理,就不會出現錯亂的問題。
這個方案只能作為一個參考方案,有一個弊端就是在列表數據多的時候,會創建很多新的對象,而占用內存。那么大家有什么更好的方案可以分享分享。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。