溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android中ListView的item按鈕監(jiān)聽錯亂問題解決辦法

發(fā)布時間:2020-07-06 19:20:20 來源:網(wǎng)絡(luò) 閱讀:2001 作者:Activity_Intent 欄目:移動開發(fā)

    在開發(fā)中經(jīng)常會遇到這樣的問題,做了一個列表,給列表的每一項添加了按鈕監(jiān)聽事件,但是在列表的數(shù)據(jù)很多的時候經(jīng)常會出現(xiàn)點(diǎn)擊后錯亂的問題。對于這種問題,我們在程序中可能都有自己的解決辦法,但是你也許第一次發(fā)現(xiàn)這個問題的時候會跟我之前一樣手足無措。

    那么現(xiàn)在我們可以分析一下這種問題的根本原因。


    首先,我們來看一下一個出錯的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();
		}

		//這里拿出來數(shù)據(jù)集合里的當(dā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出來的mStrData卻不是點(diǎn)擊的那一項
				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)用的時候,將集合中的當(dāng)前項數(shù)據(jù)拿出來付給了成員變量mStrData,程序繼續(xù)往下執(zhí)行:

viewHolder.item_button_test.setOnClickListener(new OnClickListener() {

			@Override
			public void onClick(View v) {
			        //這里toast出來的mStrData卻不是點(diǎn)擊的那一項
				Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show();
			}
		});

這里給item的按鈕添加了監(jiān)聽事件,但是要注意:程序并不會回調(diào)監(jiān)聽事件中的

@Override
public void onClick(View v) {
    //這里toast出來的mStrData卻不是點(diǎn)擊的那一項
    Toast.makeText(mContext, "您點(diǎn)擊了-" + mStrData, Toast.LENGTH_LONG).show();
}

而是會繼續(xù)回調(diào)getView方法。

等到列表即將被加載完成,也就是最后一次回調(diào)getView方法時,成員變量mStrData會被最后一次賦值,

那么,getView方法每回調(diào)一次,mStrData的值就會被重新賦一次。

然后,當(dāng)我們點(diǎn)擊按鈕,就會回調(diào)監(jiān)聽的onClick方法,這時候執(zhí)行toast:

Toast.makeText(mContext, "您點(diǎn)擊了-" + 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("點(diǎn)擊");
		// 給item的按鈕設(shè)置點(diǎn)擊監(jiān)聽,創(chuàng)建一個監(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();
		}
	}

}

這時候就不會出現(xiàn)問題。不同的地方就在于給item的button添加點(diǎn)擊事件的時候是每次都創(chuàng)建一個新的MyAdapterListener對象來實(shí)現(xiàn)這個監(jiān)聽。那么在new這個MyAdapterListener對象的時候給他傳入一個position做為標(biāo)記,這樣每一個item都會有一個屬于自己的監(jiān)聽類,我們就可以在這個監(jiān)聽類中做一些自己的邏輯處理,就不會出現(xiàn)錯亂的問題。

    這個方案只能作為一個參考方案,有一個弊端就是在列表數(shù)據(jù)多的時候,會創(chuàng)建很多新的對象,而占用內(nèi)存。那么大家有什么更好的方案可以分享分享。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI