Android中ListView的item按钮监听错乱问题解决办法

在开发中经常会遇到这样的问题,做了一个列表,给列表的每一项添加了按钮监听事件,但是在列表的数据很多的时候经常会出现点击后错乱的问题。对于这种问题,我们在程序中可能都有自己的解决办法,但是你也许第一次发现这个问题的时候会跟我之前一样手足无措。

那么现在我们可以分析一下这种问题的根本原因。

首先,我们来看一下一个出错的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都会有一个属于自己的监听类,我们就可以在这个监听类中做一些自己的逻辑处理,就不会出现错乱的问题。

这个方案只能作为一个参考方案,有一个弊端就是在列表数据多的时候,会创建很多新的对象,而占用内存。那么大家有什么更好的方案可以分享分享。

时间: 2024-10-29 19:10:41

Android中ListView的item按钮监听错乱问题解决办法的相关文章

Android中Preference的使用以及监听事件分析

> 在Android系统源码中,绝大多数应用程序的UI布局采用了Preference的布局结构,而不是我们平时在模拟器中构建应用程序时使用的View布局结构,例如,Setting模块中布局.当然,凡事都有例外,FMRadio应用程序中则使用了View布局结构(可能是该应用程序是marvel公司提供的,如果由google公司做,那可说不准).归根到底,Preference布局结构和View的布局结构本质上还是大同小异,Preference的优点在于布局界面的可控性和高效率以及可存储值的简洁性(每个

Android中Button的五种监听事件

简单聊一下Android中Button的五种监听事件: 1.在布局文件中为button添加onClick属性,Activity实现其方法2.匿名内部类作为事件监听器类3.内部类作为监听器4.Activity本身作为事件监听器,实现onClickListener5.外部类作为监听器 ButtonListenerActivity.class public class ButtonListenerActivity extends AppCompatActivity implements View.On

android listVIew实现button按钮监听程序

1.重写simpleAdapter 方法@Override public HashMap<String,String> getItem(int position) { // TODO Auto-generated method stub return listItem.get(position); } public MyAdapter(Context context, ArrayList<HashMap<String, String>> data, int resour

Android自定义ListView的Item无法响应OnItemClick的解决办法

转: 如果你的自定义ListViewItem中有Button或者Checkable的子类控件的话,那么默认focus是交给了子控件,而ListView的Item能被选中的基础是它能获取Focus,也就是说我们可以通过将ListView中Item中包含的所有控件的focusable属性设置为false,这样的话ListView的Item自动获得了Focus的权限,也就可以被选中了 我们可以通过对Item Layout的根控件设置其Android:descendantFocusability=”bl

Android 中listview的item点击事件失效了的解决办法

原因多半是由于在自己定义的Item中存在诸如ImageButton,Button,CheckBox等子控件(也可以说是Button或者Checkable的子类控件),此时这些子控件会将焦点获取到,所以常常当点击item时变化的是子控件,item本身的点击没有响应. 即在Item布局的根布局加上android:descendantFocusability=”blocksDescendants”的属性就好了

iOS中jQuery 的delegate 事件监听无效解决办法

前端的小朋友将原来 以下结构的代码 <a href="ssss"> <p>sssss</p> <p>dddddd</p> </a> 改成了 <div data-href="ssss"> <p>ssssss</p> <p>dddddd</p> </div> 因为涉及到一些异步加载的内容,所以使用如下代码来实现 a 标签的功能

android之Android中的SQL查询语句LIKE绑定参数问题解决办法(sqlite数据库)

由于考虑到数据库的安全性,不被轻易SQL注入,执行查询语句时,一般不使用直接拼接的语句,而是使用参数传递的方法.然后在使用参数传递的方法中时,发现当使用like方式查询数据时,很容易出现一个问题. 错误案例: 复制代码代码如下: String myname = "abc";String sql = "select * from mytable where name like '?%'";Cursor cursor = db.rawQuery(sql, new Str

ListView 删除item删除不了的问题解决办法

下面的方法是删除不了item的: Integer pos = Integer.valueOf(msg.getBody().toString()); adapter.getList().remove(pos); adapter.notifyDataSetChanged(); 下面的方法可以删除: Integer pos = Integer.valueOf(msg.getBody().toString()); adapter.getList().remove((int)pos); adapter.n

Android耳机线控详解,蓝牙耳机按钮监听(仿酷狗线控效果)

转载请注明出处:http://blog.csdn.net/fengyuzhengfan/article/details/46461253 当耳机的媒体按键被单击后,Android系统会发出一个广播,该广播的携带者一个Action名为MEDIA_BUTTON的Intent.监听该广播便可以获取手机的耳机媒体按键的单击事件. 在Android中有个AudioManager类,该类会维护MEDIA_BUTTON广播的分发,所以要实现耳机按键监听需要向AudioManager注册一个用于接收耳机按键单击