当前位置: 移动技术网 > IT编程>移动开发>Android > Android ListView优化之提高android应用效率

Android ListView优化之提高android应用效率

2019年07月24日  | 移动技术网IT编程  | 我要评论

龚小京背景,里约大冒险迅雷下载,年终奖个税计算公式

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();

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网