在開發(fā)中經(jīng)常會(huì)遇到這樣的問題,做了一個(gè)列表,給列表的每一項(xiàng)添加了按鈕監(jiān)聽事件,但是在列表的數(shù)據(jù)很多的時(shí)候經(jīng)常會(huì)出現(xiàn)點(diǎn)擊后錯(cuò)亂的問題。對(duì)于這種問題,我們?cè)诔绦蛑锌赡芏加凶约旱慕鉀Q辦法,但是你也許第一次發(fā)現(xiàn)這個(gè)問題的時(shí)候會(huì)跟我之前一樣手足無(wú)措。
創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括嘉魚網(wǎng)站建設(shè)、嘉魚網(wǎng)站制作、嘉魚網(wǎng)頁(yè)制作以及嘉魚網(wǎng)絡(luò)營(yíng)銷策劃等。多年來(lái),我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,嘉魚網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到嘉魚省份的部分城市,未來(lái)相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!那么現(xiàn)在我們可以分析一下這種問題的根本原因。
首先,我們來(lái)看一下一個(gè)出錯(cuò)的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 ArrayListmStrings; private LayoutInflater mInflater; private String mStrData; public MyDataAdapter(Context c, ArrayList 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(); } //這里拿出來(lái)數(shù)據(jù)集合里的當(dāng)前這一項(xiàng)mStrData mStrData = mStrings.get(position); viewHolder.item_textView_content.setText(mStrData); viewHolder.item_button_test.setText("點(diǎn)擊"); //這里給item的button設(shè)置了點(diǎn)擊監(jiān)聽事件 viewHolder.item_button_test.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //這里toast出來(lái)的mStrData卻不是點(diǎn)擊的那一項(xiàng) Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show(); } }); return convertView; } class MyViewHolder { TextView item_textView_content; Button item_button_test; } }
然后,我們分析一下原因,相信老程序員都可以看出問題的所在:
mStrData = mStrings.get(position);
getView方法第一次被調(diào)用的時(shí)候,將集合中的當(dāng)前項(xiàng)數(shù)據(jù)拿出來(lái)付給了成員變量mStrData,程序繼續(xù)往下執(zhí)行:
viewHolder.item_button_test.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //這里toast出來(lái)的mStrData卻不是點(diǎn)擊的那一項(xiàng) Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show(); } });
這里給item的按鈕添加了監(jiān)聽事件,但是要注意:程序并不會(huì)回調(diào)監(jiān)聽事件中的
@Override public void onClick(View v) { //這里toast出來(lái)的mStrData卻不是點(diǎn)擊的那一項(xiàng) Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show(); }
而是會(huì)繼續(xù)回調(diào)getView方法。
等到列表即將被加載完成,也就是最后一次回調(diào)getView方法時(shí),成員變量mStrData會(huì)被最后一次賦值,
那么,getView方法每回調(diào)一次,mStrData的值就會(huì)被重新賦一次。
然后,當(dāng)我們點(diǎn)擊按鈕,就會(huì)回調(diào)監(jiān)聽的onClick方法,這時(shí)候執(zhí)行toast:
Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show();
此時(shí)的mStrData就是只能是最后一次賦的值了,出錯(cuò)就是必然的。
那么,來(lái)看一下我的解決方法:
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 ArrayListmStrings; private LayoutInflater mInflater; private String mStrData; public MyDataAdapter(Context c, ArrayList 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("點(diǎn)擊"); // 給item的按鈕設(shè)置點(diǎn)擊監(jiān)聽,創(chuàng)建一個(gè)監(jiān)聽的實(shí)現(xiàn)類,并傳入當(dāng)前的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, "您點(diǎn)擊了-" + mStrings.get(position), Toast.LENGTH_LONG).show(); } } }
這時(shí)候就不會(huì)出現(xiàn)問題。不同的地方就在于給item的button添加點(diǎn)擊事件的時(shí)候是每次都創(chuàng)建一個(gè)新的MyAdapterListener對(duì)象來(lái)實(shí)現(xiàn)這個(gè)監(jiān)聽。那么在new這個(gè)MyAdapterListener對(duì)象的時(shí)候給他傳入一個(gè)position做為標(biāo)記,這樣每一個(gè)item都會(huì)有一個(gè)屬于自己的監(jiān)聽類,我們就可以在這個(gè)監(jiān)聽類中做一些自己的邏輯處理,就不會(huì)出現(xiàn)錯(cuò)亂的問題。
這個(gè)方案只能作為一個(gè)參考方案,有一個(gè)弊端就是在列表數(shù)據(jù)多的時(shí)候,會(huì)創(chuàng)建很多新的對(duì)象,而占用內(nèi)存。那么大家有什么更好的方案可以分享分享。
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。