标题有点长。 这篇文章主要是讲关于ListView的两个方面
如何通过复用缓存View对象解决OOM(Out Of Memory)内存溢出的异常;
通过使用ViewHolder来减少findViewById()的次数进而让ListView显示效率提高。
一般的ListView控件都能完成显示数据条目的功能,但是你在滑动的过程中就会出现OOM的异常,导致应用程序崩溃,用户的体验非常糟糕。
OOM异常图:
那么如何解决OOM异常的问题呢?
先分析一下出现异常的原因
假设一个屏幕只能显示9个ListViewitem,当滑动手指的时候,第一个item消失,第10个item要展现的时候。 我们如果不复用view 对象,上篇博客中代码会重新创建view对象进行显示,而就在此时 第1个item对象还没有进行Garbage Collector,系统进行垃圾回收是有大段的时间间隔的。 所以不断的创建新的view对象,系统就会内存溢出。 理解了原理,解决方法就简单的多了。 在getView()方法里面加入对view参数的判断,如果view对象不为空就复用以前的view对象,如果view对象为空,就重新创建view对象。
public View getView(int position, View convertView, ViewGroup parent) {
View view ;
if(convertView!=null){
view=convertView;
}else{
view=View.inflate(getApplicationContext(), R.layout.item, null);
iv_fruit=(ImageView) view.findViewById(R.id.iv_fruit);
tv_fruit_text=(TextView) view.findViewById(R.id.tv_fruit_text);
}
if(position%2==0){
iv_fruit.setImageResource(R.drawable.orange);
tv_fruit_text.setText("橘子");
tv_fruit_text.setTextSize(30l);
}
return view;
}
- -
现在解决了OOM异常的问题,接下来就是要使用句柄提高ListView显示效率了。
看上面代码中findViewById()这个方法,是利用反射来进行查找的。这个方法是非常耗时的操作,如果频繁的进行该方法,会降低ListView效率,所以需要ViewHolder来进行优化。 将控件存在ViewHolder里面,这样查找的时候就不用去布局文件里面去耗时的进行查找了。
使用ViewHolder步骤
步骤一:定义一个ViewHolder内部类
private class ViewHolder{
ImageView fruit_jpg;
TextView fruit_text;
}
步骤二:在类的成员变量位置初始化ViewHolder,然后通过view对象设置标记和获得标记。
public View getView(int position, View convertView, ViewGroup parent) {
View view ;
if(convertView!=null){
view=convertView;
holder=(ViewHolder) view.getTag();
}else{
view=View.inflate(getApplicationContext(), R.layout.item, null);
holder.fruit_jpg=(ImageView) view.findViewById(R.id.iv_fruit);
holder.fruit_text=(TextView) view.findViewById(R.id.tv_fruit_text);
view.setTag(holder);
}
if(position%2==0){
holder.fruit_jpg.setImageResource(R.drawable.orange);
holder.fruit_text.setText("橘子");
holder.fruit_text.setTextSize(30l);
}
return view;
}