首先贊下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實作