`

利用convertView优化ListView性能

阅读更多

有一段时间没更新了 主要生活上出了点状况 然后学习的速度明显下降了(在看JS什么的 太基础的没有什么好整理的)

 

**************************正文*******************************

 

这里提到的ListView只是作为一个典型代表 其实在Android中 采用类似Adapter机制的GridView等都是可以适用的 而ListView应该是用得最多的 所以就以它来举例

大家都知道 将ListView和Adapter绑定以后 其实也就是将数据源和控件显示绑定在一起 而每次需要显示ListView的时候 里面的item其实是Adapter提供的 通过的就是重要的getView()方法 而做法也是在这上面进行

 

先来看一下基本的getView写法

 

public View getView(int position, View convertView, ViewGroup parent) {
	View view = new View();

	//通过inflate等找到布局 然后findViewById等 设置各个显示的item

	return view;
}

 

而在ListView滑动的过程中 很容易就会发现每次getView被执行 都会new出一个View对象 长此以往会产生很大的消耗 特别当item中还有Bitmap等 甚至会造成OOM的错误导致程序崩溃

 

在看getView提供的参数时 可能已经注意到了 有一个参数View convertView 而这个convertView其实就是最关键的部分了 原理上讲 当ListView滑动的过程中 会有item被滑出屏幕 而不再被使用 这时候Android会回收这个条目的view 这个view也就是这里的convertView

在上面的做法中 当item1被移除屏幕的时候 我们会重新new一个View给新显示的item_new 而如果使用了这个convertView 我们其实可以复用它 这样就省去了new View的大量开销

下面就是使用convertView后的情况

 

public View getView(int position, View convertView, ViewGroup parent) {
	View view = null;
	if (convertView != null) {
	view = convertView;
	//复用了回收的view 只需要直接作内容填充的修改就好了
	} else {
	view = new Xxx(...);
	//没有供复用的view 按一般的做法新建view
	}
	return view;
}

 

这样一来 就避免了反复创建大量view的问题了

 

但是上面的仍然有缺陷 当我们的ListView中填充的item有多种形式时 比如微博中 有的item中包含图片 有的item包含视频 那么必然的 我们需要用到2种item的布局方式

此时如果只是单纯判断convert是否存在 会造成回收的view不符合你当前需要的布局 而类似转换失败出错退出

这里要提到Adapter中的另外2个方法:

public int getItemViewType(int position) {}

public int getViewTypeCount() {}

从方法名上 就可以比较明显的明白这2个的作用

下面附上一个demo代码

 

class MyAdapter extends BaseAdapter{
	Context mContext;
	LinearLayout linearLayout = null;
	LayoutInflater inflater;
	TextView tex;
	final int VIEW_TYPE = 2;
	final int TYPE_1 = 0;
	final int TYPE_2 = 1;

	public MyAdapter(Context context) {
		mContext = context;
		inflater = LayoutInflater.from(mContext);
	}

	@Override
	public int getCount() {
		return listString.size();
	}

	//每个convert view都会调用此方法,获得当前所需要的view样式
	@Override
	public int getItemViewType(int position) {
		int p = position%6;
		if(p == 0)
			return TYPE_1;
		else if(p < 3)
			return TYPE_2;
		else
			return TYPE_1;
	}

	@Override
	public int getViewTypeCount() {
		return 2;
	}

	@Override
	public Object getItem(int arg0) {
		return listString.get(arg0);
	}

	@Override
	public long getItemId(int position) {
		return position;
	}

	@Override
	public View getView(int position, View convertView, ViewGroup parent) {
		viewHolder1 holder1 = null;
		viewHolder2 holder2 = null;
		int type = getItemViewType(position);

		//无convertView,需要new出各个控件
		if(convertView == null)
		{ 
			//按当前所需的样式,确定new的布局
			switch(type)
			{
				case TYPE_1:
				convertView = inflater.inflate(R.layout.listitem1, parent, false);
				holder1 = new viewHolder1();
				holder1.textView = (TextView)convertView.findViewById(R.id.textview1);
				holder1.checkBox = (CheckBox)convertView.findViewById(R.id.checkbox);
				convertView.setTag(holder1);
				break;
				case TYPE_2:
				convertView = inflater.inflate(R.layout.listitem2, parent, false);
				holder2 = new viewHolder2();
				holder2.textView = (TextView)convertView.findViewById(R.id.textview2);
				holder2.imageView = (ImageView)convertView.findViewById(R.id.imageview);
				convertView.setTag(holder2);
				break;
			}
		}
		else
		{
			//有convertView,按样式,取得不用的布局
			switch(type)
			{
				case TYPE_1:
				holder1 = (viewHolder1) convertView.getTag();
				break;
				case TYPE_2:
				holder2 = (viewHolder2) convertView.getTag();
				break;
			}
				//设置资源
			switch(type)
			{
				case TYPE_1:
				holder1.textView.setText(Integer.toString(position));
				holder1.checkBox.setChecked(true);
				break;
				case TYPE_2:
				holder2.textView.setText(Integer.toString(position));
				holder2.imageView.setBackgroundResource(R.drawable.icon);
				break;
			}
		}
		return convertView;
	}
}
//各个布局的控件资源
class viewHolder1{
	CheckBox checkBox;
	TextView textView;
}
class viewHolder2{
	ImageView imageView;
	TextView textView;
}

 这里对于每个View使用了一个viewHolder来控制其内部的子item

还有一个需要注意的地方是使用了setTag和getTag的方法 将holder绑定到了view上 也算一种技巧

 

以上基本就是主要的内容了 下面再补充实际操作当中的一些Tips

*如果convertView上用Type区分有些繁琐 或者不需要那么复杂 只是很少有出现不同的情况 那么还可以在取得convertView后 通过java提供的 instanceof 来判断是否可以强转 如果不能强转 就去新建一个View的做法 但是其实这种做法并不规范 所以还是推荐上面的做法

*第二个是关于ListView 对于纯色的item背景 其实可以直接设置BackgroundColor 而不要使用图片 这一部分其实可以有不小的提升 同样的 对于任何纯色的背景 应该尽量去设置RGB颜色 而不是全用一张图片做背景

 

感谢guoxinya86指出 在做类型强转的时候 这里用type作为区分类型的判断 但是实际情况下 很可能出现系统回收的convertView与要创建的并不相符 所以在强转处的type判断是不保险的 考虑了下还是应该使用instanceof做一下判断 或者再为每个View绑定一个type的标记 然后再决定是重用还是重新创建

1
1
分享到:
评论
7 楼 deux9876 2014-08-04  
为什么position%6(6怎么来的)。
还有,为什么判断p==0后,还要判断p是否小于3(不能是小于5或者大于1吗?)
6 楼 cookiejj2010 2012-04-20  
guoxinya86 写道
嗯,这样会比较保险!

:) 非常感谢指出 当时也没看仔细
5 楼 guoxinya86 2012-04-19  
嗯,这样会比较保险!
4 楼 cookiejj2010 2012-04-19  
guoxinya86 写道
楼主想看看,有没有可能存在convertView是TYPE_1类型的,而你强转成TYPE_2类型

看了一下 在强转之前判断了type的类型 但是似乎的确这里type的值和convertView似乎会不一样 保险一点的话 还是用instanceof判断一下吧 或者再自己为每个view绑定一个type的值 然后根据这个条件去创建新的或者重用
3 楼 guoxinya86 2012-04-19  
楼主想看看,有没有可能存在convertView是TYPE_1类型的,而你强转成TYPE_2类型
2 楼 cookiejj2010 2012-04-18  
guoxinya86 写道
楼主你好,针对你的代码有个问题想跟你讨论下,您没有判断当前convertView是什么类型的,你那样直接强转会不会有问题啊?

这里已经事先根据type分类了
int type = getItemViewType(position);
可以再getItemViewType里去确定类型 所以强转应该不会有问题 另外也可以用instanceof来判断一下
1 楼 guoxinya86 2012-04-18  
楼主你好,针对你的代码有个问题想跟你讨论下,您没有判断当前convertView是什么类型的,你那样直接强转会不会有问题啊?

相关推荐

    listView中多个listItem布局时,convertView缓存及使用

    listView中多个listItem布局时,convertView缓存及使用

    listview 图片异步加载 图片错位 软引用 getView调用次数 convertView重用

    * listview的第一个元素(A)是公用一个convertView的(一般情况),此时问题就来了,如果异步下载图片执行的比较慢,第一个元素对应的url * 下载的图片会放到第一个元素上面还是滑动后第一个可见的元素上面呢? ...

    Android ListView多种布局优化demo

    Android ListView多种布局优化demo,使用了两种优化手段,包括convertView,ViewHolder,对应的我的博客地址是: http://blog.csdn.net/u012320459/article/details/47667869

    Listview加载的性能优化是如何实现的

    listview加载的核心是其adapter,本文针对listview加载的性能优化就是对adpter的优化,总共分四个层次: 0、最原始的加载 1、利用convertView 2、利用ViewHolder 3、实现局部刷新 〇、最原始的加载 这里是不经...

    ListView的单一布局优化demo

    使用了convertView和viewHolder两种优化方式,ListView的子布局是单一布局,对应的我的博客地址是: http://blog.csdn.net/u012320459/article/details/47667869

    ListView(乱跳问题完美(包括点击闪动)解决-每一个条目都做动画-不复用convertview20160402)

    1. ListView(乱跳问题完美(包括点击闪动)解决_每一个条目都做动画_不复用convertview20160402) 动画默认显示3行,展开显示全部。 自己在公司做项目,遇到这个问题,就写个类似的demo。供大家相互学习参考,有不足之...

    ListView(不乱跳但是会出现点击闪动-不复用convertview20160402)

    不复用,listview就不会乱跳 .因为listview条目中的控件有做动画效果(展开与关闭)。 .但是此时,点击展开与关闭,效果正常。但是滑动出屏幕以后全部关闭了。也属于正常状态。 .也就是说,只能对当前显示的做动画,...

    ListView多条目显示

    ListView复用多个布局文件。 //复用三个布局文件 @Override public View getView(int position, View convertView, ViewGroup parent) { viewHolder1 holder1 = null; viewHolder2 holder2 = null; ...

    android ListView滑动分页加载和点击分页加载,以及复用convertView综合案例

    android ListView滑动分页加载和点击分页加载,以及复用convertView综合案例

    Android ListView常见的优化方式详解

    其实listview的工作原理就是,listview在请求屏幕可见的item数时,convertView在getVIew中是null 的. 但是当屏幕向下滑动的时候(比如该屏幕尺寸可显示7条teim),在item1被隐藏,此时出现item8时,covertView的值就不为...

    fragment下的listview实现

    fragment下的listview实现,其中LISTVIEW有效数据()返回的数据,convertView不优化,效果类似滑动条.

    ListView控件

    实现 ListView 局部刷新 刷新 public View getView(int position, View convertView, ViewGroup parent) { ViewHold viewHold = null; final int position1 = position; if(null == convertView){ viewHold...

    android 之listview 优化方法

    1、listview在定义的时候宽和高最好固定一下,这样可以减少listview测量次数,避免每次加载的时候都要去进行测量。 2、分页加载,这也是优化之一,优化内存,还有体验感,有兴趣的可以试试大数据加载; 3、...

    ListView的多种子布局优化demo

    ListView的多种子布局优化demo,使用了convertView和viewHolder两种优化方式,对应的我的博客地址是: http://blog.csdn.net/u012320459/article/details/47667869

    listviewAdapter

    public View getView(int position, View convertView, ViewGroup parent) { // TODO Auto-generated method stub ViewHolder viewHolder = null; if (convertView == null) { convertView = inflater.inflate...

    Android里ListView里面添加CheckBox实现多选

    Android里ListView里面添加CheckBox实现多选,用上ViewHolder和ConvertView

    ListVeiw适配器工具类+焦点抢占+convertView复用导致内容错乱

    ListVeiw适配器工具类+焦点抢占+convertView复用导致内容错乱

    listview优化方法

    第一种优化就是重用convertView,这也是最简单的一种优化方式,就是在Adapter类的getView方法中通过判断convertView是否为null,是的话就需要在创建一个视图出来,然后给视图设置数据,最后将这个视图返回给底层,...

    ListView异步创建View

    异步创建View这种操作一般情况下是用不...以往的我们使用一个Listview一般都是为了展示一类布局相同的信息,这种情况下,我们可以通过adapter的getView()方法中的convertView来实现View的复用,使View不用反复创建。

    ListView异步加载图片

    ①ListView异步加载图片的方式 ②给ImageView设置Tag,解决图片覆盖问题 ③采用LruCache缓存已经加载过的图片 ④当ListView滚动时不加载图片,滚动停止时才加载图片,从而达到ListView滑动很流畅的效果 ⑤当...

Global site tag (gtag.js) - Google Analytics