1. ViewHolder
感觉ViewHolder就不用多说了吧,这是ListView最基本的优化技巧了。ViewHolder机制使在getView()中避免了每次都要进行findViewById()去实例化控件,通过视图缓存机制重用缓存即可。
在后面ListView加载不同布局中也用到了ViewHolder机制,所以示例代码就不单独贴了。
2. 不要在getView()中进行耗时操作
(1)在getView()中进行耗时操作会造成卡顿,因此诸如加载图片这一类耗时操作,需要用异步的方式处理耗时任务。
(2)还有一点就是要控制异步任务的执行频率,因为当用户频繁的上下滑动,会瞬间产生上百个异步任务,会带来无意义的大量的UI更新操作,因此可以考虑在列表滑动时停止进行异步任务,直到列表停下来。
//判断列表的状态 public void onScrollStateChange(AbsLiatView view, int scrollState){ if(scrollState == OnScrollListener.SCROLL_STATE_IDLE){ mIsViewIdle = true; }else{ mIsViewIdle = false; } } //在getView()中根据列表状态选择在静止时加载图片 if(mIsViewIdle = true){ imageView.setTag(url); //通过url异步加载图片到imageView }
3. 开启硬件加速
从Android3.0开始,Android的2D显示管道被被设计得更加支持硬加速了,硬加速使用GPU承担了所有在View的canvas上执行的绘制操作。有些莫名其妙的卡顿现象可以通过为Activity开启硬件加速来优化,在Manifest中的<activity>标签下进行如下设置。
andorid:hardwareAccelerated=”true”
还可以在运行时使用以下代码禁止个别的View的硬加速,因为硬件加速可能在自定义View出问题。
if(myView.isHardwareAccelerated()){ myView.setLayerType(View.LAYER_TYPE_SOFTWARE,null); }
4. 如何实现ListView中具有不同的布局
我之前做过一个聊天界面,每个ListView的Item都具有A方和B方的所有布局信息,只是在合适的时候hide掉某一方而已,但是后来发现了一个机智的官方也考虑到了这种情况,并推出了两个方法,专门来解决这个问题,这里总结一下。
@Override public int getItemViewType(int position) { return type; } @Override public int getViewTypeCount() { return number; }
上面两个方法,getItemViewType()用来返回第position个条目是何种类型,而getViewTypeCount()则返回不同布局的总数。结合这两种方法,即可在getView中进行灵活判断加载不同的布局了。
下面是实现简单的聊天对话的例子,主要是ListView的部分:
public class ChatItemListViewAdapter extends BaseAdapter { private List<ChatItemListViewBean> mData; private LayoutInflater mInflater; public ChatItemListViewAdapter(Context context, List<ChatItemListViewBean> data) { this.mData = data; mInflater = LayoutInflater.from(context); } @Override public int getCount() { return mData.size(); } @Override public Object getItem(int position) { return mData.get(position); } @Override public long getItemId(int position) { return position; } @Override public int getItemViewType(int position) { ChatItemListViewBean bean = mData.get(position); return bean.getType(); } @Override public int getViewTypeCount() { return 2; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder; if (convertView == null) { if (getItemViewType(position) == 0) { holder = new ViewHolder(); convertView = mInflater.inflate(R.layout.chat_item_itemin, null); holder.icon = (ImageView) convertView.findViewById(R.id.icon_in); holder.text = (TextView) convertView.findViewById(R.id.text_in); } else { holder = new ViewHolder(); convertView = mInflater.inflate(R.layout.chat_item_itemout, null); holder.icon = (ImageView) convertView.findViewById(R.id.icon_out); holder.text = (TextView) convertView.findViewById(R.id.text_out); } convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } holder.icon.setImageBitmap(mData.get(position).getIcon()); holder.text.setText(mData.get(position).getText()); return convertView; } public final class ViewHolder { public ImageView icon; public TextView text; } }
其中getViewTypeCount()返回了布局总数为两种,getItemViewType()则实时返回某条目的类型,从而在getView中加载不同的布局,类型信息使用Bean类存储,并在Activity中进行数据的初始化。其中可以看到ViewHolder内部类的应用,在getView中首先会判断convertView是否为空,若为空才去重新加载布局,否则使用缓存。
效果图如下所示,源码地址点击下载。最后在下二篇Android开发——ListView使用技巧总结(二)中介绍一下如何在合适的时候隐藏Toolbar,还有如何实现具有弹性的ListView。