首先赞下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中的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<View>();
} 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 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;
} @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<User>(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<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);
} @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<Integer, Object>

    SparseBooleanArray 用于替代形如  HashMap<Integer, Boolean>

    SparseIntArray 用于替代形如  HashMap<Integer, Integer>

    SparseLongArray 用于替代形如  HashMap<Integer, Long>

support.v4.util.SparseArrayCompat 提供了v4包相应平台的 SparseArray实现

Android BaseAdapter和ViewHolder 优化 解决ListView的item抢焦点问题和item错乱问题的更多相关文章

  1. Android中利用ViewHolder优化自定义Adapter的典型写法

    利用ViewHolder优化自定义Adapter的典型写法 最近写Adapter写得多了,慢慢就熟悉了. 用ViewHolder,主要是进行一些性能优化,减少一些不必要的重复操作.(WXD同学教我的. ...

  2. android Baseadapter 和 ViewHolder的使用

    昨晚学习了徐大神的关于BaseAdapter的讲解,让我受益匪浅特来博客留下印记 说到baseadapter大家一定都不陌生,下面这张图就展示了数据.listview.baseadapter 之间的关 ...

  3. 解决ListView在界面只显示一个item

    ListView只显示一条都是scrollview嵌套listView造成的,将listView的高度设置为固定高度之后,三个条目虽然都完全显示.但是这个地方是动态显示的,不能写死.故采用遍历各个子条 ...

  4. Ubuntu中Android SDK Manager无法更新解决办法

    Ubuntu中Android SDK Manager无法更新解决办法http://hi.baidu.com/petercao2008/item/d7a64441f04668e81e19bc1a

  5. Android内存优化解决 资料和总结的经验分享

    在前公司做一个图片处理的应用时, 项目交付的时候,客户的手机在运行应用的时候,一直在崩溃,而这个异常就是OutOfMemory的错误,简称为OOM, 搞得我们也是极其的崩溃,最后 ,我们是通过网上搜集 ...

  6. [Android]对BaseAdapter中ViewHolder编写简化

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/3642849.html 在Android项目中,经常都会用到Li ...

  7. android 自定义adapter和线程结合 + ListView中按钮滑动后状态丢失解决办法

    adapter+线程 1.很多时候自定义adapter的数据都是来源于服务器的,所以在获取服务器的时候就需要异步获取,这里就需要开线程了(线程池)去获取服务器的数据了.但这样有的时候adapter的中 ...

  8. [Android]对BaseAdapter中ViewHolder编写简化(转)

    来自博客:http://www.cnblogs.com/tiantianbyconan/p/3642849.html 在Android项目中,经常都会用到ListView这个控件,而相应的Adapte ...

  9. Android开发之漫漫长途 XIV——ListView

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

随机推荐

  1. Poj1733 Parity Game(带权并查集)

    题面 Poj 题解 反正只要你判断是否满足区间的奇偶性,假设每一位要么是\(1\)要么是\(0\)好了. 假设有\(S\)的前缀和为\(sum[]\),则有: 若\(S[l...r]\)中有奇数个\( ...

  2. Flask实战第56天:板块管理

    cms布局 编辑 cms_boards.html {% block main_content %} <div class="top-box"> <button c ...

  3. ACM的奇计淫巧_bitset优化

    什么是bitset bitset 是STL库中的二进制容器,根据C++ reference 的说法,bitset可以看作bool数组,但优化了空间复杂度和时间复杂度,并且可以像整形一样按位与或. 使用 ...

  4. 【DFS】XIII Open Championship of Y.Kupala Grodno SU Grodno, Saturday, April 29, 2017 Problem D. Divisibility Game

    题意:给你一个序列,长度不超过52,每个元素不超过13.让你重新对这个序列排序,sum(i)表示i的前缀和,使得排序过后,对每个i,都有sum(i)%i==0. 深搜,加两个优化:①倒着从后向前搜:② ...

  5. 【构造】Gym - 100781B - Bell Ringing

    根据n-1推n的情况,让n从每一个的最右走到最左,再从下一个最左走到最右,如此往复即可. #include<cstdio> using namespace std; int n,a[500 ...

  6. bzoj 1636: [Usaco2007 Jan]Balanced Lineup -- 线段树

    1636: [Usaco2007 Jan]Balanced Lineup Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 772  Solved: 560线 ...

  7. 使用参数化查询防止SQL注入漏洞

    参数化查询防止SQL注入漏洞 看别人的登录注册sql语句有没漏洞即可 Where  name=‘admin’ or ‘1=1’ and password=’123’; 可以Or ‘1=1’就是漏洞 h ...

  8. [Linux] Proc 文件系统

    转载自:http://linux.chinaunix.net/doc/2004-10-05/16.shtml#324lfindex0 目录: /proc --- 一个虚拟文件系统 加载 proc 文件 ...

  9. Java9模块化(Jigsaw)初识

    Java9经历了多次跳票,终于要在9月份正式发布,原计划Jigsaw在Java7就有的,也终于在Java9里面提供了,简单总结下. 对比 Java9 以前 上面2个图分别对应的分别是JDK8/9的目录 ...

  10. java性能监控器VisualVM

    1.mac系统启动文件位置:/Library/Java/JavaVirtualMachines/jdk1.8.0_101.jdk/Contents/Home/bin/jvisualvm 2.linux ...