Android BaseAdapter和ViewHolder 优化 解决ListView的item抢焦点问题和item错乱问题
首先贊下hyman大神
曾經僅僅是簡單的重寫個BaseAdapter,將getView方法保持抽象。而ViewHolder沒有抽象過。
。。
ViewHolder (用了一個集合+泛型管理存取view)
/*** author : stone* email : aa86799@163.com* time : 15/7/24 14 27*/ public class StoneViewHolder {private int mPosition;private View mConvertView;private SparseArray<View> mViews; //管理listView-item中的viewpublic 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<View>();}public View getConvertView() {return mConvertView;}public static StoneViewHolder getInstance(Context context, int layoutId, int position, ViewconvertView, ViewGroup parent) {if (convertView == null) {return new StoneViewHolder(context, layoutId, position, parent);} else {StoneViewHolder holder = (StoneViewHolder) convertView.getTag();holder.mPosition = position; //更新復用的convertView中 positionreturn holder;}}public <T extends View> T getView(int viewId) {View view = mViews.get(viewId);if (view == null) {view = mConvertView.findViewById(viewId);mViews.put(viewId, view);}return (T) view;}public <T> void setTag(int viewId, T tag) {getView(viewId).setTag(tag);}public <T> T getTag(int viewId) {return (T) getView(viewId).getTag();}/*------------------------ 設置view屬性(以后擴展) --------------------------------*/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 /*** author : stone* email : aa86799@163.com* time : 15/7/24 14 46*/ public abstract class StoneListAdapter<T> extends BaseAdapter {private List<T> mData;private Context mContext;private int mLayoutID;public StoneListAdapter(Context context, int layoutID, List<T> data) {this.mContext = context;this.mLayoutID = layoutID;this.mData = data == null ? new ArrayList<T>() : data;}@Overridepublic int getCount() {return mData.size();}@Overridepublic T getItem(int position) {return mData.get(position);}@Overridepublic long getItemId(int position) {return position;}@Overridepublic 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<User>(ListViewActivity.this, R.layout.activity_listview_item, mData) {@Overrideprotected 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() {@Overridepublic void onClick(View v) {}});} };
?假設 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<User> {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);}@Overrideprotected void getView(Context context, final StoneViewHolder holder, final int position) {CheckBox cb = holder.getView(R.id.cb_check);cb.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {@Overridepublic 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<Integer, Object>
? ? SparseBooleanArray 用于替代形如 ?HashMap<Integer, Boolean>
? ? SparseIntArray 用于替代形如 ?HashMap<Integer, Integer>
? ? SparseLongArray 用于替代形如 ?HashMap<Integer, Long>
? support.v4.util.SparseArrayCompat 提供了v4包相應平臺的 SparseArray實現
轉載于:https://www.cnblogs.com/yfceshi/p/6786035.html
總結
以上是生活随笔為你收集整理的Android BaseAdapter和ViewHolder 优化 解决ListView的item抢焦点问题和item错乱问题的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: LeetCode | Minimum D
- 下一篇: 业务活动图