天天看點

android listview item 錯位,Android BaseAdapter和ViewHolder 優化 解決ListView的item搶焦點問題和item錯亂問題...

首先贊下hyman大神

曾經僅僅是簡單的重寫個BaseAdapter,将getView方法保持抽象。而ViewHolder沒有抽象過。

。。

ViewHolder (用了一個集合+泛型管理存取view)

public class StoneViewHolder {

private int mPosition;

private View mConvertView;

private SparseArray mViews; //管理listView-item中的view

public StoneViewHolder(Context context, int layoutId, int position, ViewGroup parent) {

this.mPosition = position;

this.mConvertView = LayoutInflater.from(context).inflate(layoutId, parent, false);

this.mConvertView.setTag(this);

this.mViews = new SparseArray();

}

public View getConvertView() {

return mConvertView;

}

public static StoneViewHolder getInstance(Context context, int layoutId, int position, View

convertView, ViewGroup parent) {

if (convertView == null) {

return new StoneViewHolder(context, layoutId, position, parent);

} else {

StoneViewHolder holder = (StoneViewHolder) convertView.getTag();

holder.mPosition = position; //更新複用的convertView中 position

return holder;

}

}

public T getView(int viewId) {

View view = mViews.get(viewId);

if (view == null) {

view = mConvertView.findViewById(viewId);

mViews.put(viewId, view);

}

return (T) view;

}

public void setTag(int viewId, T tag) {

getView(viewId).setTag(tag);

}

public T getTag(int viewId) {

return (T) getView(viewId).getTag();

}

public StoneViewHolder setText(int viewId, String text) {

((TextView)getView(viewId)).setText(text);

return this;

}

public StoneViewHolder setText(int viewId, int resId) {//R.string.

((TextView)getView(viewId)).setText(resId);

return this;

}

public StoneViewHolder setImageBitmap(int viewId, Bitmap bitmap) {

((ImageView)getView(viewId)).setImageBitmap(bitmap);

return this;

}

public StoneViewHolder setImageResource(int viewId, int resId) {

((ImageView)getView(viewId)).setImageResource(resId);

return this;

}

}

Adapter

public abstract class StoneListAdapter extends BaseAdapter {

private List mData;

private Context mContext;

private int mLayoutID;

public StoneListAdapter(Context context, int layoutID, List data) {

this.mContext = context;

this.mLayoutID = layoutID;

this.mData = data == null ? new ArrayList() : data;

}

@Override

public int getCount() {

return mData.size();

}

@Override

public T getItem(int position) {

return mData.get(position);

}

@Override

public long getItemId(int position) {

return position;

}

@Override

public View getView(int position, View convertView, ViewGroup parent) {

StoneViewHolder holder = StoneViewHolder.getInstance(mContext, mLayoutID, position,

convertView, parent);

getView(mContext, holder, position);

return holder.getConvertView();

}

protected abstract void getView(Context context, StoneViewHolder holder, int position);

}

在ListViewActivity中使用

stoneBaseAdapter = new StoneListAdapter(ListViewActivity.this, R.layout.activity_listview_item, mData) {

@Override

protected void getView(Context context, final StoneViewHolder holder, final int position) {

User user = getItem(position);

holder.setText(R.id.tv_id, user.getId()).setText(R.id.tv_name, user.getName())

.setText(R.id.tv_age, user.getAge() + "");

holder.getView(R.id.btn_test).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

}

});

}

};

關于Adapter中View搶焦點:

假設 listView.setOnItemClickListener(listener);   且item中的  button.setOnClickListener(listener);

無論怎麼點選,button會一直被觸發...

僅僅須要在item的root-layout中 加入 一個屬性:   android:descendantFocusability="blocksDescendants"

關于item-view複用後。顯示混亂:

有時條目過多,滑動到下一屏資料時。有些view複用後,view的狀态(比方CheckBox的選種狀态。ImageView的圖檔反複出現)會變亂。

一般處理呢。須要有一個機制,來管理一種相應關系: 目前position相應哪種狀态

比方說checkBox選中狀态混亂:

class MyAdapter extends StoneListAdapter {

private SparseBooleanArray mCheckStateArray;

public MyAdapter(Context context, int layoutID, List data) {

super(context, layoutID, data);

this.mCheckStateArray = new SparseBooleanArray();

}

public void setChecked(int position, boolean isChecked) {

mCheckStateArray.put(position, isChecked);

}

public boolean isChecked(int position) {

return mCheckStateArray.get(position);

}

@Override

protected void getView(Context context, final StoneViewHolder holder, final int position) {

CheckBox cb = holder.getView(R.id.cb_check);

cb.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {

@Override

public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {

setChecked(position, isChecked);//記錄狀态。防緩存顯示

}

});

cb.setChecked(isChecked(position));

}

}

關于SparseArray

SparseArray 内部實作是Array數組。當長度不夠時,會調用System.arrayCopy

内部有 keys和values兩個數組。

put(key, value); 二分法查找key應該存放的位置  由于key是Integer類型

put、get時 兩個數組都是操作的同一個位置上的資料

SparseArray 用于替代形如  HashMap

SparseBooleanArray 用于替代形如  HashMap

SparseIntArray 用于替代形如  HashMap

SparseLongArray 用于替代形如  HashMap

support.v4.util.SparseArrayCompat 提供了v4包相應平台的 SparseArray實作