Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android中ListView的item按鈕監聽錯亂問題解決辦法

Android中ListView的item按鈕監聽錯亂問題解決辦法

編輯:關於Android編程

在開發中經常會遇到這樣的問題,做了一個列表,給列表的每一項添加了按鈕監聽事件,但是在列表的數據很多的時候經常會出現點擊後錯亂的問題。對於這種問題,我們在程序中可能都有自己的解決辦法,但是你也許第一次發現這個問題的時候會跟我之前一樣手足無措。     那麼現在我們可以分析一下這種問題的根本原因。       首先,我們來看一下一個出錯的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: 1 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都會有一個屬於自己的監聽類,我們就可以在這個監聽類中做一些自己的邏輯處理,就不會出現錯亂的問題。     這個方案只能作為一個參考方案,有一個弊端就是在列表數據多的時候,會創建很多新的對象,而占用內存。那麼大家有什麼更好的方案可以分享分享。
  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved