一般是针对包含多个元素的View,如ListView,GridView,ExpandableListview,的时候我们是给其设置一个Adapter。Adapter是与View之间提供数据的桥梁,也是提供每个Item的视图桥梁。

以ListView为例,其工作原理为:

● ListView针对List中每个item, adapter都会调用一个getView的方法获得布局视图

●我们一般会Inflate一个新的View,填充数据并返回显示

当然如果我们的Item很多话(比如上万个),都会新建一个View吗?很明显这样内存是接受不了的,Google也不会这么做,Android中有个叫做Recycler的构件,下图是他的工作原理:

很明显,无论数据中是多少个item,在显示上Recycler只存储其中可见的View在内存中。当向下滑动时,顶部不可见Item直接回移动到下方再次填充数据变为新增项。这样就不用每次都新建一个View了。

这个也就是我们在Adapter中常见的getView方法的调用,对应此方法我们就能看出,convertView就是每一Item在Recyler之前的布局视图。

  • public View getView(int position, View convertView, ViewGrouppare

所以,Android已经给我们提供了Recycler机制了,我们就应该利用此机制,而不是每次都去inflate一个View。

Example

Don’t

  1. public View getView(int position, View convertView, ViewGroupparent){
  2. convertView = LayoutInflater.from(mContext).inflate(R.layout.item_view,null);
  3. //dosomething…
  4. return converView;
  5. }

Do

  1. public View getView(int position, View convertView, ViewGroupparent){
  2. if (convertView ==null) {
  3. convertView =LayoutInflater.from(mContext).inflate(R.layout.item_view, null);
  4. }
  5. //dosomething…
  6. return converView;
  7. }

ViewHolder的作用

之前所说的Recycler模式是为了解决重复inflate时候造成的View资源浪费,还哪有什么方法何可再次优化我们的性能吗?答案是Yes。

我们还是从getView中的每一个方法调用去查看,发现其实我们拿到convertView的时候,每次都会根据这个布局去findViewById。如下,使我们通常的写法:

findViewById是在解析layout.xml布局那种其中的子View,解析xml是一个力气活,所以Google也建议我们将这个费力不讨好的活优化起来,所以提出了ViewHolder的概念。

即,使用一个静态类,保存xml中的各个子View的引用关系,这样就不必要每次都去解析xml了。如下:就是针对上面代码写的一个ViewHolder

  1. if (convertView == null) {
  2. convertView = mInflater.inflate(R.layout.item_view, null);
  3. }
  4. TextView titleTextView = (TextView) convertView.findViewById(R.id.text));
  5. ImageView iconImageView = (ImageView)convertView.findViewButId( R.id.icon));
  6. //DoSomething…

findViewById是在解析layout.xml布局那种其中的子View,解析xml是一个力气活,所以Google也建议我们将这个费力不讨好的活优化起来,所以提出了ViewHolder的概念。

即,使用一个静态类,保存xml中的各个子View的引用关系,这样就不必要每次都去解析xml了。如下:就是针对上面代码写的一个ViewHolder

  1. static class ViewHolder {
  2. TextView titleTextView;
  3. ImageView iconImageView;
  4. }

但是,在getView方法中我们只能拿到三个参数,position、convertView、viewGroup是拿不到我们自定义的ViewHolder的。所以,我们希望通过convertView拿到ViewHolder只能将其放在tag里。

下面是一个完整的ViewHolder使用exmaple:

  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. ViewHolder holder;
  3. if (convertView == null) {
  4. convertView = mInflater.inflate(R.layout.item_view, null);
  5. holder = new ViewHolder();
  6. holder.titleTextView = (TextView) convertView.findViewById(R.id.text);
  7. holder.iconImageView = (ImageView) convertView.findViewById(R.id.icon);
  8. convertView.setTag(holder);
  9. } else {
  10. holder = (ViewHolder) convertView.getTag();
  11. }
  12. holder.titleTextView.setText(DATA[pos].title);
  13. holder.iconImageView.setImageBitmap(DATA[pos].bitmap);
  14. return convertView;
  15. }
  16. static class ViewHolder {
  17. TextView titleTextView;
  18. ImageView iconImageView;
  19. }

Tips. Support.v7中的RecyclerView 就是采用了此思想来制作的。

多个类型的ViewType

当我们在Adapter中调用方法getView的时候,如果整个列表中的Item View如果有多种类型布局,如:

我们继续使用convertView来将数据从新填充貌似不可行了,因为每次返回的convertView类型都不一样,无法重用。

Android在设计上的时候,也想到了这点。所以,在adapter中预留的两个方法。

  • public int getItemViewType(int position) ;
  • public int getViewTypeCount();

只需要重新这两个方法,设置一下ItemViewType的个数和判断方法,Recycler就能有选择性的给出不同的convertView了。

Example:

  1. @Override
  2. public intgetItemViewType(int position) {
  3. if (DATA[pos].type == 0) {
  4. return 0;
  5. } else {
  6. return 1;
  7. }
  8. }
  9. @Override
  10. public int getViewTypeCount() {
  11. return 2;
  12. }
  13. @Override
  14. public View getView(int position, View convertView, ViewGroup arg2) {
  15. TitleViewHolder titleHolder;
  16. InfoViewHolder infoHolder;
  17. int type = getItemViewType(position);
  18. if (convertView == null) {
  19. switch (type) {
  20. case 0:
  21. convertView = mInflater.inflate(R.layout.item_view, null);
  22. titleHolder = new TitleViewHolder();
  23. titleHolder.titleTextView = (TextView) convertView.findViewById(R.id.text);
  24. titleHolder.iconImageView = (ImageView) convertView.findViewById(R.id.icon);
  25. convertView.setTag(titleHolder);
  26. break;
  27. case 1:
  28. convertView = mInflater.inflate(R.layout.item_view2, null);
  29. infoHolder = new InfoViewHolder();
  30. infoHolder.titleTextView = (TextView) convertView.findViewById(R.id.text);
  31. convertView.setTag(infoHolder);
  32. break;
  33. }
  34. } else {
  35. switch (type) {
  36. case 0:
  37. titleHolder = (TitleViewHolder) convertView.getTag();
  38. break;
  39. case 1:
  40. infoHolder = (InfoViewHolder) convertView.getTag();
  41. break;
  42. }
  43. }
  44. switch (type) {
  45. case 0:
  46. titleHolder.titleTextView.setText(DATA[pos].title);
  47. break;
  48. case 1:
  49. infoHolder.titleTextView.setText(DATA[pos].title);
  50. infoHolder.iconImageView.setImageBitmap(DATA[pos].bitmap);
  51. break;
  52. }
  53. return convertView;
  54. }
  55. static class TitleViewHolder {
  56. public ImageView iconImageView;
  57. public TextView titleTextView;
  58. }
  59. static class InfoViewHolder {
  60. TextView titleTextView;
  61. ImageView iconImageView;
  62. }

NotifyDataSetChanged刷新机制

当ListView中的数据发生了改变,我们希望刷新ListView中的View时,我们一般会调用NotifyDataSetChanged来刷新ListView。看一下它的源码:

  1. public void notifyChanged() {
  2. synchronized (mObservers) {
  3. // 向每一个子View发送onChanged
  4. for (int i = mObservers.size() - 1; i >= 0; i--) {
  5. mObservers.get(i).onChanged();
  6. }
  7. }
  8. }

发 现它针对每一个子View都做了刷新,当然,如果我们的数据都变量还可以理解。但是,一般条件下,我们需要更新的View不多。频繁的调用 NotifyDataSetChanged方法,刷新整个界面不合适。这样会把界面上显示的所有item都全部重绘一次,即使只有一个view的内容发生 了变化。

所以,我们可以写一个update的方法,来单独刷新一个View

  1. private void updateView(int itemIndex){
  2. intvisiblePosition = yourListView.getFirstVisiblePosition();
  3. Viewv = yourListView.getChildAt(itemIndex - visiblePosition);
  4. ViewHolder viewHolder =(ViewHolder)v.getTag();
  5. if(viewHolder!= null){
  6. viewHolder.titleTextView.setText("我更新了");
  7. }
  8. }

Adapter中的网络图片优化

ListView中的每一项Item基本都会带着网络图片,当item比较多的时候,过多的网络请求和过多的图片存储都会是ListView变慢变卡。

所以针对其做一下优化:

●  采用线程池进行网络图片请求,网络图片请求获取后使用本地缓存处理(LRUCache),内存+本地文件缓存。当然,为了防止内存溢出与回收不及时,需要使用弱引用(WeakReference)来存储内存中的图片。

●  对网络中取到的图片进行按比例缩放,以减少内存消耗。

●  滑动的时候不需要对网络图片进行请求。因为,网络请求一般比较耗时,某Item的图片,在请求来的时候如果被Recycler换掉,图片就会对应不上该Item。

Tips.网络请求的工具类比较多不方便举例子,但是使用比较频繁的网络图片请求工具类就是Volley了,Volley提供了一个ImageLoader的工具类和NetworkImageView的网络图片请求View

本文链接:http://www.eoeandroid.com/thread-536377-1-1.html

【编辑推荐】

【责任编辑:chenqingxiang TEL:(010)68476606】

Android高手进阶:Adapter深入理解与优化的更多相关文章

  1. Android高手进阶——Adapter深入理解与优化

    Android高手进阶--Adapter深入理解与优化 通常是针对包括多个元素的View,如ListView,GridView.ExpandableListview,的时候我们是给其设置一个Adapt ...

  2. Android 高手进阶之自定义View,自定义属性(带进度的圆形进度条)

      Android 高手进阶(21)  版权声明:本文为博主原创文章,未经博主允许不得转载. 转载请注明地址:http://blog.csdn.net/xiaanming/article/detail ...

  3. Android高手进阶教程(二十八)之---Android ViewPager控件的使用(基于ViewPager的横向相册)!!!

      分类: Android高手进阶 Android基础教程 2012-09-14 18:10 29759人阅读 评论(35) 收藏 举报 android相册layoutobjectclassloade ...

  4. Android高手进阶教程(五)之----Android 中LayoutInflater的使用!

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://weizhulin.blog.51cto.com/1556324/311450 大 ...

  5. Android高手进阶教程(七)之----Android 中Preferences的使用!

    http://blog.csdn.net/Android_Tutor/article/details/5531849 大家好,我们这一节讲的是Android Preferences 的学习,Prefe ...

  6. Android 高手进阶,自己定义圆形进度条

    背景介绍 在Android 开发中,我们常常遇到各种各样绚丽的控件,所以,依靠我们Android本身所带的控件是远远不够的,许多时候须要我们自定义控件,在开发的过程中.我们公司遇到了一种须要自己写的一 ...

  7. Android高手进阶教程(十七)之---Android中Intent传递对象的两种方法(Serializable,Parcelable)!

    [转][原文] 大家好,好久不见,今天要给大家讲一下Android中Intent中如何传递对象,就我目前所知道的有两种方法,一种是Bundle.putSerializable(Key,Object); ...

  8. Android高手进阶篇4-实现侧滑菜单框架,一分钟集成到项目中

    先来看下面的这张效果图: 上面这张效果图是百度影音的,现在在Android上很流行,最初是由facebook自己实现的,而后各大应用有跟风之势,那么这种侧滑效果是如何实现的呢? 网上现在这种侧滑菜单的 ...

  9. Android Adapter基本理解

    感谢大佬:https://blog.csdn.net/l799069596/article/details/47301711 Android Adapter基本理解: 我的理解是: 1.一个有许多ge ...

随机推荐

  1. JS函数的定义与调用方法

    JS函数调用的四种方法:方法调用模式,函数调用模式,构造器调用模式,apply,call调用模式 1.方法调用模式:先定义一个对象,然后在对象的属性中定义方法,通过myobject.property来 ...

  2. HRESULT 0x80131515 解决方法

    http://254698001.blog.51cto.com/2521548/1339696 很多朋友在加载DLL是发生这样的错误:HRESULT: 0x80131515. 解决方法是,在DLL文件 ...

  3. Python 存储模型

    1.Python彻底分离了对象和引用,可以认为内存中的对象都是不可修改的,每次修改引用,相当于在堆上重新创建一个对象,引用指向新对象. 2.对于数值和字符串,修改意味着引用指向一个新对象. 3.集合中 ...

  4. C++ foreach

    考虑下面的需求,对vector<int>中的每个元素加1,如何做? void add(int& lhs) // 注意:要修改主调方法中的数据,这里要使用引用 { lhs= lhs ...

  5. 项目优化经验分享(八)TeamLeader经验总结

    引言 通过前面的七篇博客.我把自己在项目优化过程的经验进行了分享,今天这篇博客,作为一个总结,就来讲讲作为一个TeamLeader,在项目管理中遇到的问题和解决经验! 正文 问题一:团队之间怎么沟通? ...

  6. swift 获取UI上某点点颜色

    extension UIView { func colorOfPoint (point: CGPoint) -> UIColor { var pixel = UnsafePointer<C ...

  7. as3.0 interface接口使用方法

    [转]as3.0 interface接口使用方法 AS在2.0的时候就支持接口了 接口能够让你的程序更具扩展性和灵活性,打个例如 比方你定义了一个方法 代码: public function aMet ...

  8. SQLServer恢复表级数据

    最近几天,公司的技术维护人员频繁让我恢复数据库,因为他们总是少了where条件,导致update.delete出现了无法恢复的后果,加上那些库都是几十G.恢复起来少说也要十几分钟.为此,找了一些资料和 ...

  9. io cache

    http://blog.163.com/digoal@126/blog/static/163877040201571511020418 http://dirlt.com/

  10. 给指针malloc分配空间后就等于数组吗?

    首先回答这个的问题:严格的说不等于数组,但是可以认为它是个数组一样的使用而不产生任何问题.不过既然这样,那它应该算是个数组吧.所以,一般我们都用“动态数组”这种名字来称呼这种东西. 要讲清楚这个东西, ...