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

通常是针对包括多个元素的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, ViewGroupparent)

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

Example

Don’t

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

Do

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

ViewHolder的作用

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

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

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

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

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

static class ViewHolder {
TextView titleTextView;
ImageView iconImageView;
}

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

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

    public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if (convertView == null) {
convertView = mInflater.inflate(R.layout.item_view, null);
holder = new ViewHolder();
holder.titleTextView = (TextView) convertView.findViewById(R.id.text);
holder.iconImageView = (ImageView) convertView.findViewById(R.id.icon);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
holder.titleTextView.setText(DATA[pos].title);
holder.iconImageView.setImageBitmap(DATA[pos].bitmap);
return convertView;
} static class ViewHolder {
TextView titleTextView;
ImageView iconImageView;
}

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:

    @Override
public intgetItemViewType(int position) {
if (DATA[pos].type == 0) {
return 0;
} else {
return 1;
}
} @Override
public int getViewTypeCount() {
return 2;
} @Override
public View getView(int position, View convertView, ViewGroup arg2) {
TitleViewHolder titleHolder;
InfoViewHolder infoHolder;
int type = getItemViewType(position); if (convertView == null) {
switch (type) {
case 0:
convertView = mInflater.inflate(R.layout.item_view, null);
titleHolder = new TitleViewHolder();
titleHolder.titleTextView = (TextView) convertView.findViewById(R.id.text);
titleHolder.iconImageView = (ImageView) convertView.findViewById(R.id.icon);
convertView.setTag(titleHolder);
break;
case 1:
convertView = mInflater.inflate(R.layout.item_view2, null);
infoHolder = new InfoViewHolder();
infoHolder.titleTextView = (TextView) convertView.findViewById(R.id.text);
convertView.setTag(infoHolder);
break;
}
} else {
switch (type) {
case 0:
titleHolder = (TitleViewHolder) convertView.getTag();
break;
case 1:
infoHolder = (InfoViewHolder) convertView.getTag();
break;
}
}
switch (type) {
case 0:
titleHolder.titleTextView.setText(DATA[pos].title);
break;
case 1:
infoHolder.titleTextView.setText(DATA[pos].title);
infoHolder.iconImageView.setImageBitmap(DATA[pos].bitmap);
break;
} return convertView;
} static class TitleViewHolder {
public ImageView iconImageView;
public TextView titleTextView;
} static class InfoViewHolder {
TextView titleTextView;
ImageView iconImageView;
}

NotifyDataSetChanged刷新机制

当ListView中的数据发生了改变。我们希望刷新ListView中的View时,我们通常会调用NotifyDataSetChanged来刷新ListView。

看一下它的源代码:

    public void notifyChanged() {
synchronized (mObservers) {
// 向每个子View发送onChanged
for (int i = mObservers.size() - 1; i >= 0; i--) {
mObservers.get(i).onChanged();
}
}
}

发现它针对每个子View都做了刷新,当然,假设我们的数据都变量还能够理解。可是。一般条件下,我们须要更新的View不多。

频繁的调用NotifyDataSetChanged方法。刷新整个界面不合适。这样会把界面上显示的所有item都所有重绘一次,即使仅仅有一个view的内容发生了变化。

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

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

Adapter中的网络图片优化

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

所以针对其做一下优化:

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

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

●  滑动的时候不须要对网络图片进行请求。

由于,网络请求一般比較耗时。某Item的图片,在请求来的时候假设被Recycler换掉。图片就会相应不上该Item。

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

/**

 * @author zhoushengtao(周圣韬)

 * @since 2014年7月8日 下午15:08:29

 * @weixin stchou_zst 

* @blog http://blog.csdn.net/yzzst

 */

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

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

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

  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. 如何使用银联卡充值美元到BTC-E以及比特币搬砖教程

    1,名词解释 搬砖:就是在价格低的平台买入比特币,然后转移到价格高的平台卖出, 一般而言,BTC-E是国外三大比特币交易所中比特币单价最低的一个站,因为其需要用美元充值,相对不方便.之前国内比特币价格 ...

  2. qq邮箱是怎么做到同一个浏览器让多个不用用户同时打开的? --session的控制

    待解:..... 借鉴网址:http://www.zhihu.com/question/20235500 欢迎来讨论.....

  3. 正态分布(Normal distribution)又名高斯分布(Gaussian distribution)

    正态分布(Normal distribution)又名高斯分布(Gaussian distribution),是一个在数学.物理及project等领域都很重要的概率分布,在统计学的很多方面有着重大的影 ...

  4. 2008r2 显示桌面图标

  5. SICP练习1.6-1.8

    1.6 死循环 1.7 #lang racket (define (square x) (* x x)) (define (sqrt-iter guess x) (if (good-enough? g ...

  6. Python中打印列表的序号和内容

    ==>the start 最近作业里要用到遍历打印出列表中的序号和内容,我刚开始用了个很笨的方法来写,后来老师说可以使用enumerate()函数,所以我就特意研究了下. 先看我之前用的笨方法: ...

  7. python读取文件内容方法

    1) readline 每次读一行,返回序列 2) readlines 一次全部读出,返回序列 3) numpy 的genfromtxt,返回为np的矩阵格式 import numpy as np f ...

  8. CSS实现强制换行-------Day 78

    事实上最早的时候也考虑过这个问题,当时还在想须要判定文字的长度么,实在是傻到极点了,原来CSS中本来就有这个样式设置的.而今天正好看到了有这么一篇介绍.细致看了下,感觉还不错,这里也把实验的结果记录下 ...

  9. 路径中“/” "\" "\\"的区别

    Unix使用斜杆/ 作为路径分隔符,而web应用最新使用在Unix系统上面,所以目前所有的网络地址都采用 斜杆/ 作为分隔符. Windows由于使用 斜杆/ 作为DOS命令提示符的参数标志了,为了不 ...

  10. UVA 11292 - The Dragon of Loowater (water)

    http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=24&page=sh ...