ListView优化
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>的方法缓存,不过不好用了,现在大多都用
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对象,做耗时操作
ListView优化相关推荐
- ListView优化的代码
第三种ListView优化:通过convertView+ViewHolder来实现,ViewHolder就是一个静态类,使用 ViewHolder 的关键好处是缓存了显示数据的视图(View),加快了 ...
- ListView优化机制及滑动时数据时出现的数据错乱重复问题
ListView优化机制及滑动时数据时出现的数据错乱重复问题 转载于:https://www.cnblogs.com/zhujiabin/p/5596998.html
- 【提升栈】ListView优化
前言 ListView优化一直是一个老生常谈的问题.无论是面试还是寻常的开发中,ListView永远不会被忽略掉,那么这篇文章我们来看看怎样最大化的优化ListView的性能. 1.在adapter中 ...
- Android之ListView优化
关于ListView几个方面的优化: ListView的大小设定固定值; 复用convertView, 使用ViewHolder提高在容器中查找组件的效率; 使用分页加载; 快速滚动时, item不显 ...
- 面试之 listview优化
2019独角兽企业重金招聘Python工程师标准>>> 摘用别人的,谢谢作者 Adapter的作用是界面与数据之间的桥梁,通过设置适配器至ListView控件后(如调用ListVie ...
- Android listview优化以及listview混合itemview
listView = (ListView) findViewById(R.id.listview); mAdapter = new VideoListAdapter(this); listView.s ...
- Android优化系列之ListView优化老生常谈
本文内容:adapter,listview的优化,RecycleBi,google大会推荐优化, 实现ListView的过程,Adapter起到了至关重要的作用,不仅仅因为getview()方法.那么 ...
- ListView优化机制及滑动时数据时出现的数据错乱重复问题 图片、checkBox等
该篇内容主要是记录我在实际开发中遇到的ListView滑动时数据错乱的几种情况,以及解决方法.在进行ListView滑动时数据错乱问题讨论之前会对ListView所谓的<优化>进行说明.文 ...
- Android ListView优化之局部刷新(更新)(非notifyDataSetChanged)
转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53020164 [DylanAndroid的csdn博客] 在Android开发 ...
- Android开发实战《手机安全卫士》——8.“通信卫士”模块实现 JUnit测试 ListView优化
文章目录 1.高级工具--去电归属地显示 2.通信卫士--黑名单布局编写 3.通信卫士--黑名单数据库 4.通信卫士--黑名单CRUD功能实现 5.通信卫士--JUnit测试 6.通信卫士--黑名单号 ...
最新文章
- Android深入浅出系列之Android工具的使用—模拟器(一)
- 线程池中使用条件变量和信号量的性能比较
- 飞桨 第一课 传统图像识别是怎么做的+Aistudio python数据可视化2020.3.31;2020.4.2补
- GitHub 发布重磅更新:你电脑上的 IDE 可以删了?!
- LayaBox IDE 安装后执行项目报错解决方案的一些记录
- canvas html 动态,canvas.html
- 好用的下拉第三方——nicespinner
- springboot 接口404_资深架构带你学习Springboot集成普罗米修斯
- Python基础项目实践之:学生信息管理系统
- 对python文件方法open的探究
- SpringBoot报错couldn‘t check if tables are already present using metadata:
- jdk1.5新特性5之枚举之模拟枚举类型
- silverlight将字符串转化为控件
- python卡尔曼滤波_卡尔曼滤波+单目标追踪+python-opencv
- SpringBoot快速开始magic-api基础
- 如何正确学习和使用设计模式
- python写入TXT文件乱码,且utf-8无效
- linux操作系统2试题,linux操作系统试题
- 0基础学编程需要哪些基础?
- 习题 9.10 已有a,b两个链表,每个链表中的结点包括学号、成绩。要求把两个链表合并,按学号升序排列。