Android ListView优化之提高android应用效率
listview是一个经常用到的控件,listview里面的每个子项item可以使一个字符串,也可以是一个组合控件。adapter是listview和数据源间的中间人。
当每条数据进入可见区域时,adapter的getview()会被调用,返回代表具体数据的视图。触摸滚动时,频繁调用。支持成百上千条数据。
下面为显示每条数据的xml文件:
<linearlayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal"> <imageview android:id="@+id/icon" android:layout_width="48dip" android:layout_height="48dip" /> <textview android:id="@+id/text" android:layout_gravity="center_vertical" android:layout_width="0dip" android:layout_weight="1.0" android:layout_height="wrap_content" /> </linearlayout>
1。最简单的方法,最慢且最不实用
public view getview(int pos, view convertview, viewgroup parent){ view item = minflater.inflate(r.layout.list_item, null); ((textview) item.findviewbyid(r.id.text)). settext(data[pos]); ((imageview) item.findviewbutid(r.id.icon)). setimagebitmap((pos & 1) == 1 ? micon1 : micon2); return item; }
2。利用convertview回收视图,效率提高200%。
public view getview(int pos, view convertview, viewgroup parent){ if (convertview == null) { convertview = minflater.inflate( r.layout.list_item, null); } ((textview) convertview.findviewbyid(r.id.text)). settext(data[pos]); ((imageview) convertview.findviewbutid(r.id.icon)). setimagebitmap((pos & 1) == 1 ? micon1 : micon2); return convertview; }
3。利用viewholder模式,效率在提高50%
static class viewholder { textview text; imageview icon; } public view getview(int pos, view convertview, viewgroup parent){ viewholder holder; if (convertview == null) { convertview = minflater.inflate(r.layout.list_item, null); holder = new viewholder(); holder.text = (textview) convertview.findviewbyid( r.id.text)); holder.icon = (imageview) convertview.findviewbutid( r.id.icon)); convertview.settag(holder); } else { holder = (viewholder) convertview.gettag(); } holder.text.settext(data[pos]); holder.icon.setimagebitmap((pos & 1) == 1 ? micon1 : micon2); return convertview; }
adapter更新效率比较:
1的更新不到10 frames/second
2的更新接近30 frames/second
3的更新接近40 frames/second
背景和图像
视图背景图像总会填充整个视图区域
1。图像尺寸不合适会导致自动缩放
2。避免实时缩放
3。最好预先缩放到视图大小
originalimage = bitmap.createscaledbitmap( originalimage, // ????缩放图像 view.getwidth(), // 视图宽度 view.getheight(), // 视图高度 true); // ????线性过滤器
1的效率接近25 frames/second
2的效率接近50 frames/second
默认情况下, 窗口有一个不透明的背景
有时可以不需要
-????????最高层的视图是不透明的
- ???? 最高层的视图覆盖整个窗口
layout_width = fill_parent layout_height = fill_parent
更新看不见的背景是浪费时间
删除窗口背景:
1。修改编码
public void oncreate(bundle icicle){ super.oncreate(icicle); setcontentview(r.layout.mainview); // 删除窗口背景 getwindow().setbackgrounddrawable(null); ... }
2。修改xml
首先确定你的res/values/styles.xml有
<resources> <style name="nobackgroundtheme" parent="android:theme"> <item name="android:windowbackground">@null</item> </style> </resources>
然后编辑androidmainfest.xml
<activity android:name="myapplication" android:theme="@style/nobackgroundtheme"> ... </activity>
更新请求
当屏幕需要更新时,调用invalidate()方法,简单方便,但是更新了整个视图,代价太高。
最好先找到无效区域,然后调用
invalidate(rect dirty); invalidate(int left, int top, int right, int bottom);
视图和布局
如果一个窗口包含很多视图,启动太慢,绘制时间长,用户界面反应速度很慢
解决方法:
1。使用textview的复合drawable减少层次
<textview android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/hello" android:drawableleft="@drawable/icon"/>
2。使用viewstuf延迟展开视图
在xml文件中定义viewstuf
<viewstub android:id = "@+id/stub_import" android:inflatedid="@+id/panel_import" android:layout="@layout/progress_overlay" android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_gravity="bottom"/>
在需要展开视图时,
findviewbyid(r.id.stub_import).setvisibility(view.visible); // 或者 view importpanel = ((viewstub) findviewbyid(r.id.stub_import)).inflate();
3。使用<merge>合并中间视图
默认情况下,布局文件的根作为一个节点,加入到父视图中,如果使用merge可以避免根节点
<merge xmlns:android = "http://schemas.android.com/apk/res/android"> <! -- content --> </merge>
4。使用ralativelayout减少层次
<relativelayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="wrap_content"> <imageview android:id="@+id/icon" android:layout_width="48dip" android:layout_height="48dip" android:layout_alignparentleft="true" android:layout_centervertical="true"/> <textview android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/text_line1" android:layout_alignparenttop="true" android:layout_torightof="@id/icon"/> <textview android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/text_line2" android:layout_torightof="@id/icon" android:layout_below="@id/text_line1"/> <checkbox android:id="@+id/star" android:layout_width="48dip" android:layout_height="48dip" android:layout_alignparentright="true" android:layout_centervertical="true"/> </relativelayout>
5.使用自定义视图
class customview extends view { @override protected void ondraw(canvas canvas) { // 加入你的绘图编码 } @override protected void onmeasure(int widthmeasurespec, int heightmeasurespec) { // 计算视图的尺寸 setmeasureddimension(widthspecsize, heightspecsize); } }
6 使用自定义布局
class gridlayout extends viewgroup { @override protected void onlayout(boolean changed, int l, int t, int r, int b) { final int count = getchildcount(); for (int i=0; i < count; i++) { final view child = getchildat(i); if (child.getvisibility() != gone) { // 计算子视图的位置 child.layout(left, top, right, bottom); } } } }
内存分配
在性能敏感的代码里,避免创建java对象
1。测量 onmeasure()
2。布局onlayout()
3。绘图 ondraw() dispatchdraw()
4。事件处理 ontouchevent() dispatchtouchevent()
5。adapter: getview() bindview()
强行限制(适用调试模式)
int prevlimit = -1; try { prevlimit = debug.setallocationlimit(0); // 执行不分配内存的代码 } catch (dalvik.system.allocationlimiterror e) { // 如果代码分配内存, java 虚拟机会抛出错误 log.e(logtag, e); } finally { debug.setallocationlimit(prevlimit); }
管理好对象:
1。适用软引用:内存缓存的最佳选择
2。适用弱引用:避免内存泄露
内存缓存:
private final hashmap<string, softreference<t>> mcache; public void put(string key, t value) { mcache.put(key, new softreference<t>(value)); } public t get(string key, valuebuilder builder) { t value = null; softreferece<t> reference = mcache.get(key); if (reference != null) { value = reference.get();