- 浏览: 375202 次
- 性别:
- 来自: 上海
文章分类
最新评论
-
liqiong0327:
解决了困扰我2天的问题。
Unity3D中 Android插件findviewbyid返回null的解决办法 -
liqiong0327:
大神啊,给跪了!!!!
Unity3D中 Android插件findviewbyid返回null的解决办法 -
wangzhongquan_:
Chrome中canvas上drawImage无法画出image的解决办法 -
deux9876:
为什么position%6(6怎么来的)。还有,为什么判断p= ...
利用convertView优化ListView性能 -
wuweitiandian:
请假一下~如果我不是通过new Images()来加入图片,而 ...
Chrome中canvas上drawImage无法画出image的解决办法
有一段时间没更新了 主要生活上出了点状况 然后学习的速度明显下降了(在看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的标记 然后再决定是重用还是重新创建
评论
还有,为什么判断p==0后,还要判断p是否小于3(不能是小于5或者大于1吗?)
:) 非常感谢指出 当时也没看仔细
看了一下 在强转之前判断了type的类型 但是似乎的确这里type的值和convertView似乎会不一样 保险一点的话 还是用instanceof判断一下吧 或者再自己为每个view绑定一个type的值 然后根据这个条件去创建新的或者重用
这里已经事先根据type分类了
int type = getItemViewType(position);
可以再getItemViewType里去确定类型 所以强转应该不会有问题 另外也可以用instanceof来判断一下
发表评论
-
Unity3D中 Android插件findviewbyid返回null的解决办法
2013-12-31 10:58 4347最近在给Unity3D接入各个Android ... -
WebView判断手势缩放
2012-09-05 11:42 4906wv.setOnTouchListener(new OnT ... -
Android重启应用程序代码
2012-06-14 10:12 26610很久没有更新,换了家公司,在做手机端的网游,采取的 ... -
打开自己程序下的html文件资源并显示的方法
2012-02-10 16:48 4255在工作中碰到一个情 ... -
Java中的软引用,弱引用和虚引用
2012-01-17 14:38 2061在Android的图片处理中, ... -
[转载]Android实现ListView异步加载图片
2012-01-11 16:25 1540原文分享地址:http://apps.hi.baidu. ... -
Mars视频笔记——Oauth认证(三)
2012-01-04 11:15 1589有了上面Oauth原理,HTTP操作,AsyncTask的基础 ... -
Mars视频笔记——HTTP操作1,2,3
2012-01-04 11:11 13224-7 HTTP操作(一) 1 什么是HTTP协议 ... -
Mars视频笔记——异步操作AsyncTask
2012-01-04 11:09 17174-5 异步操作(一) 1 什么是线程的阻塞 2 什 ... -
Mars视频笔记——Oauth认证原理
2012-01-04 10:56 2449想看一下AsyncTack的使用方法 记得写过什么 但是找了半 ... -
startActivityForResult 和 startActivity
2011-12-08 16:32 2095首先说下这个问题的情景: 当前的软件有一个主Activit ... -
Activity中创建Dialog的注意事项(改变Dialog中的message)
2011-09-15 16:00 2000在写一个小游戏的过程中碰到这样一个情况 通过onCreateD ... -
Android中手势监听GestureDetector基础
2011-09-15 11:51 2982最近做的2个东西中都碰到了需要对用户手势输入监听并做出一定的 ... -
Android 同步歌词滚动 唱片手势滑动 原理手记
2011-09-08 14:32 3225刚才误删了一次。。。还好有个打开的页面没关掉 先上效果图 ... -
Android试手——Dota铃声
2011-08-23 15:41 1030手滑先发到博客频道了。。不知道现在这样算不算重复发帖。。 ... -
Android试手——Dota铃声
2011-08-23 15:27 1277前段时间在论坛上看到一个帖子,说了一个关于短信dota铃 ... -
ListView长按获取item的方法
2011-08-12 14:33 2520在做Notes的过程中碰到一个问题 我为ListView的每 ... -
Mars视频笔记——JSON解析
2011-08-04 16:39 1852JSON数据解析 其实关于json在做毕业设计的时候就 ... -
Mars视频笔记——Animation(4)
2011-08-03 14:44 1501Animations使用(4) LayoutAnima ... -
Mars视频笔记——Animation(3)
2011-08-02 16:12 3336Animations的使用(3) 1 Anima ...
相关推荐
这篇博客“ListView性能优化系列之一”着重探讨了如何通过深入理解源码和利用特定工具来提升ListView的性能。 首先,了解ListView的工作原理是优化的基础。ListView采用的是滚动复用机制,即“视图回收”...
因此,对ListView进行性能优化至关重要。本文将详细介绍几个关键的优化策略。 1. 使用ViewHolder模式 ViewHolder模式是ListView优化的核心技巧,它可以避免在滚动时频繁地创建和查找视图对象。通过创建一个...
1. **ViewHolder模式**:这是ListView性能优化的核心。通过复用已创建的视图,避免频繁的findViewById操作,减少内存分配,提升列表滑动的流畅性。在getView方法中,通过convertView参数检查并重用已存在的视图,而...
首先,ViewHolder模式是ListView性能优化的基础。这个模式的主要目的是减少对视图的查找和更新操作,通过复用已创建的视图来避免每次滚动时创建新的视图对象。在`getView()`方法中,我们首先检查convertView是否为...
本篇文章将深入探讨ListView与BaseAdapter的工作原理,并详细介绍如何进行性能优化。 首先,我们来理解一下Adapter在Android中的作用。Adapter是连接数据源和UI组件的桥梁,它负责将数据转化为可显示的视图。在...
1. **使用ViewHolder模式**:这是ListView优化的核心,通过复用convertView来减少 findViewById 的次数,提高性能。ViewHolder类存储了ListView项布局中的各个视图引用,当需要创建新视图时,直接从ViewHolder中获取...
本篇将主要探讨ListView的优化策略,以及如何利用`SwipeRefreshLayout`增强用户体验。 首先,ListView的优化主要涉及以下几个方面: 1. **视图复用**:ListView通过`convertView`机制实现视图复用,减少因创建新...
【ListView性能优化】是Android开发中的重要话题,因为ListView作为展示大量数据的组件,其性能直接影响到用户体验。以下是一些关键的优化策略: 1. **数据适配性能优化**: - **convertView重用**:ListView的...
5. **ListView的复用机制**:ListView的视图复用是其核心优化之一,通过convertView参数在`getView()`方法中重用已创建的View,避免为每个列表项创建新的对象,减少内存开销。 6. **延时加载(Lazy Loading)**:只...
以上就是ListView的几种优化策略,通过这些方法,我们可以显著提升ListView的性能,使其在处理大量数据时仍能保持流畅的滚动体验。在实际开发中,根据项目的具体需求,还可以结合其他优化手段,如使用RecyclerView...
通过以上策略,我们可以有效地优化ListView中网络图片的加载和缓存,提升应用的性能和用户体验。在实际开发中,应根据项目需求和资源选择合适的优化方案,并结合源代码分析和调试,确保优化效果达到预期。
总之,优化ListView需要从Adapter的getView方法入手,合理利用convertView和ViewHolder,减少View的创建和查找,提高滚动时的流畅性。这些优化技巧不仅适用于ListView,也适用于其他基于AdapterView的组件,如...
总结,优化ListView的关键在于视图复用,同时合理处理点击事件,美化界面,并根据需要自定义对话框和输入方式。通过这些技巧,可以提升用户体验,降低应用的资源消耗。希望以上分享能对你的开发工作有所帮助。
总结来说,优化ListView主要在于提高滚动时的流畅性,减少不必要的资源消耗,而分类显示则需要灵活运用Adapter和数据结构,创建不同类型的View来展示不同的数据。这两者结合,可以让用户在使用应用时有更好的体验。...
本篇文章将详细探讨如何通过自定义View来优化ListView,提高用户体验。 首先,理解ListView的工作原理至关重要。ListView采用的是滚动时按需加载的机制,即“视图复用”(View Recycling)。这意味着只有当前屏幕...
使用Android Profiler检查内存、CPU使用情况,找出性能瓶颈,优化ListView的性能。 通过以上这些优化技巧,我们可以让ListView在处理大量数据时保持流畅,同时减少对系统资源的占用。在实际开发中,应根据项目需求...
本文将详细介绍ListView的使用技巧,包括如何优化性能、拓展功能以及常见应用场景。 一、ListView的基本使用 1. 布局文件:在XML布局文件中,我们需要创建一个ListView,并设置其ID,例如:`android:id="@+id/...
最后,ListViewDemo可能是包含以上知识点的一个实践项目,通过这个项目,开发者可以学习如何实现这些功能,并理解在实际开发中如何应用和优化ListView。在实际操作中,一定要注意性能优化,确保用户体验的顺畅。 ...
本文将深入探讨一些优化ListView性能的方法。 首先,一个关键的优化策略是在Adapter的`getView()`方法中减少逻辑运算。`getView()`方法会在每个列表项被绘制或复用时调用,因此应避免在此处执行耗时的操作。例如,...
在Android开发中,ListView是一个非常常见的组件,用于展示大量数据的列表形式。然而,随着数据量的增加,...总之,优化ListView的关键在于合理利用View的复用机制,减少不必要的资源消耗,从而提供更好的用户体验。