Android性能优化-ListView
ListView的优化主要分为以下几点
1 convertView
的复用
ListView每次滚动都会调用getView
方法,所以优化getVieiw
是重中之重
convertView介绍
convertView
是刚刚滚动出可见区域的View的引用,此时它已经不可见,所以应该被复用以减少View的创建
优化代码
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
}else{//否则
view = convertView;//就使用可以复用的View
}
优化原因
LayoutInflater.inflate(resourceId,View)
这个方法是用来通过pull的解析方式从XML文件生成一个View对象的,如果有成千上万
个Viwe都要去解析XML生成View,会非常消耗性能
2ViewHolder
的使用
优化代码
ViewHolder viewHolder = null;
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
viewHolder = new ViewHolder();
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
viewHolder.resourceViewName = view.findViewById(resouceViewId);//从XML中找到对应的View
view.setTag(viewHolder);//将ViewHolder设置在当前ItemView的tag里面
}else{//否则
view = convertView;//就使用可以复用的View
viewHolder = (ViewHolder)convertView.getTag();//从复用的View中取出viewHoder
}
viewHolder
class ViewHolder {
TextView name;
}
优化原因
findViewById这个方法是从ViewGroup的子View里面循环遍历找id与给出的ID相同的子View,还是比较耗时的,
/**ViewGroup的FindViewByID源码*/
/**
* {@hide}
*/
@Override
protected <T extends View> T findViewTraversal(@IdRes int id) {
if (id == mID) {
return (T) this;
}
final View[] where = mChildren;
final int len = mChildrenCount;
for (int i = 0; i < len; i++) {
View v = where[i];
if ((v.mPrivateFlags & PFLAG_IS_ROOT_NAMESPACE) == 0) {
v = v.findViewById(id);
if (v != null) {
return (T) v;
}
}
}
return null;
}
3图片"三级缓存"加载优化
阐述
图片加载顺序,应该为,内存--本地--网络
1、内存缓存 优先加载,速度最快
2、本地缓存 次优先加载 速度稍快
3、网络缓存 最后加载 速度由网络速度决定(浪费流量)
代码(缓存到本地,从网络获取就不写了)
主要写一下缓存到内存中的方法,
据说以前使用HashMap<String,SoftReference<BitMap>>的方法缓存,不过不好用了,现在大多都用
LruCache,
public class MemoryCache {
private LruCache<String,Bitmap> mLruCache = null;
public MemoryCache(){
long maxMemory = Runtime.getRuntime().maxMemory();//最大内存 默认是16M
mLruCache = new LruCache<String,Bitmap>((int)(maxMemory/8)){
@Override
protected int sizeOf(String key, Bitmap value) {
//int byteCount = value.getByteCount();
//得到图片的字节数
int byteCount = value.getRowBytes() * value.getWidth();
return byteCount;
}
};
}
//从内存获取
public Bitmap getFromMemory(String url){
return mLruCache.get(url);
}
//缓存到内存
public void setToMemory(String url,Bitmap bitmap){
mLruCache.put(url,bitmap);
}
}
优化原因
从网络加载图片或者本地加载图片都比较耗时,加上Android16ms的刷新UI频率,会造成卡顿
从内存获取速度相对较快,以上只是放入内存的方法,当然压缩什么的就没有写,只是简单介绍存入内存的原理
4图片加载再次优化
导论
很多情况下ListView需要加载显示网络图片,我们尽量不要在ListView滑动的时候加载网络图片,
那样会使ListView变得卡顿所以我们要监听ListView的状态,如果ListView滑动(SCROLL_STATE_TOUCH_SCROLL
)
或者猛滑(SCROLL_STATE_FLING
)的时候,停止加载图片,否则加载图片
优化代码
listView.setOnScrollListener(new AbsListView.OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {//list停止滚动时加载图片
loadImage(startPos, endPos);// 异步加载图片 ,只加载可以看到的图片
}
}
@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
//设置当前屏幕显示的起始pos和结束pos
startPos = firstVisibleItem;
endPos = firstVisibleItem + visibleItemCount;
if (endPos >= totalItemCount) {
endPos = totalItemCount - 1;
}
}
});
优化原因
从用户的角度讲,快速滑动的时候,用户不需要看到当前内容
5 onClickListener处理
导论
有时候出了onItemClickListener
之外我们还会用到Item上其他位置的点击事件
一般情况下我们是在getView方法中,一个一个设置,就像
holder.img.setOnClickListener(new onClickListener());
这样每个都设置了一个新的OnClickListener对象,不太好
优化方案
直接在ViewHolder中设置一个position,然后viewHolder implements OnClickListener
class ViewHolder implements OnClickListener{
int position;
TextView name;
public void setPosition(int position){
this.position = position;
}
@Override
public void onClick(View v) {
switch (v.getId()){
//XXXX
}
}
}
然后再getView中设置的时候设置自己就行了
ViewHolder viewHolder = null;
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
viewHolder = new ViewHolder();
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
viewHolder.resourceViewName = view.findViewById(resouceViewId);//从XML中找到对应的View
viewHolder.setPosition(position);//设置位置
viewHolder.name.setOnClickListener(viewHolder);//设置ClickListener
view.setTag(viewHolder);//将ViewHolder设置在当前ItemView的tag里面
}else{//否则
view = convertView;//就使用可以复用的View
viewHolder = (ViewHolder)convertView.getTag();//从复用的View中取出viewHoder
}
6 总结
总之,宗旨就是少在getView里面new对象,做耗时操作