`

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

 
阅读更多

转载: http://mobile.51cto.com/abased-445617.htm

 

一般是针对包含多个元素的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, ViewGroup parent)

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

 

Example

Don’t

public View getView(int position, View convertView, ViewGroup parent){   
        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。如下,使我们通常的写法:

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

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

 

    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

 

 

  • 大小: 29.9 KB
  • 大小: 160.8 KB
  • 大小: 13.8 KB
分享到:
评论

相关推荐

    android高手进阶教程 完整版 pdf

    ### Android高手进阶教程知识点概览 #### 一、Android常用命令集锦 - **ADB命令**: ADB(Android Debug Bridge)是Android平台下用于调试的工具,它可以帮助开发者进行设备管理、应用安装与卸载等操作。 - `adb ...

    Android开发进阶从小工到专家(pdf书签版)

    《Android开发进阶从小工到专家》是一本专为Android开发者设计的进阶教程,旨在帮助初学者和有一定基础的开发者提升技能,成为一名精通Android开发的专家。这本书涵盖了Android开发的各个方面,包括基础概念、核心...

    Android开发进阶从小工到专家

    这些组件的理解和熟练运用是进阶Android开发的关键。 此外,本书还会详细介绍Android的资源管理和权限控制,这是保证应用稳定性和用户体验的重要环节。读者会学习如何有效地管理应用资源,如图片、音频和布局文件,...

    Android:View&Adapter

    在Android开发中,View和Adapter是两个至关重要的概念,它们共同构建了用户界面与数据之间的桥梁。本篇文章将深入探讨这两个组件以及它们在ListView、GridView和Spinner中的应用。 首先,让我们了解一下View。在...

    android之各种Adapter加载数据

    本文将深入探讨Android的Adapter机制,以及如何使用各种Adapter来加载数据。 首先,我们需要理解Adapter的基本概念。Adapter是Android系统中用于连接数据集(如数组、列表等)和可滚动视图(如ListView、GridView、...

    Android中ListView+Adapter

    总之,理解并熟练掌握ListView与Adapter的使用是Android开发中的重要技能。ArrayAdapter适合简单的数据展示,SimpleAdapter能处理稍微复杂的结构,而BaseAdapter则提供了最大的灵活性,适用于各种定制需求。通过实践...

    android 一个对adapter的封装使用

    在Android开发中,Adapter是一个非常重要的组件,它起到了数据与视图之间的桥梁作用。Adapter类通常用于将数据源(如数组或列表)映射到UI组件,如ListView、GridView等。在这个“android 一个对adapter的封装使用”...

    android adapter详解

    今天,我们将深入探讨 Android Adapter 的原理和实现方式。 什么是 Android Adapter? Android Adapter 是一个接口,负责将数据绑定到用户界面中。它是 Android 中一个重要的组件,广泛应用于 ListView、GridView...

    Android listview与adapter用法

    - 实例化Adapter:根据需求选择适合的Adapter类型,并传入上下文、列表项布局资源ID以及数据源。 - 绑定Adapter到ListView:使用ListView的`setAdapter()`方法将Adapter与ListView关联。 5. **自定义Adapter** -...

    android-recyclerview-v7-28.0.0

    《深入解析Android RecyclerView-v7-28.0.0组件》 RecyclerView是Android平台上的一个核心组件,自Android Lollipop(API级别21)引入以来,它已经成为了开发者构建高效、可滚动列表视图的首选工具。在Android的v7...

    android UI进阶之实现listview中checkbox的多选与记录

    本篇文章将深入探讨如何在ListView中实现CheckBox的多选功能,并记录用户的选定状态。ListView是Android中常用的一个控件,它允许我们以列表的形式展示大量数据,而CheckBox则提供了一种让用户进行多选操作的方式。 ...

    Android自定义Adapter适配器

    在Android开发中,Adapter是一个至关重要的组件,它作为数据源与UI控件之间的桥梁,使得数据能够被适配并显示在界面中。本项目聚焦于自定义Adapter的实现,特别是如何将其应用于ListView的数据显示。 首先,理解...

    android开发Adapter详解

    ### Android开发Adapter详解 #### 一、Adapter概念与作用 在Android开发中,Adapter是一种用于连接数据源和视图组件的重要桥梁。它主要用于处理数据集合与列表视图之间的交互,确保视图能够正确且高效地展示数据。...

    Android自定义adapter的listview

    Android提供了一些预定义的Adapter,如ArrayAdapter和CursorAdapter,但它们可能无法满足所有需求,因此我们常常需要自定义Adapter。 自定义Adapter的步骤如下: 1. 创建一个新的类,继承自BaseAdapter或已有的...

    Android代码-XAdapter

    >compile 'com.ydevelop:rv-adapter:0.0.9 example mRecyclerView.adapter = xRecyclerViewAdapter.apply { dataContainer = mainBeen loadMoreView = LoadMoreView(applicationContext) refreshView = ...

    Android课程设计:学生信息系统源码.zip

    1. **项目结构**:Android项目一般遵循特定的文件夹结构,如`app`、`build`、`gradle`等。`app`文件夹包含了应用的主要代码,包括`src`(源代码)、`res`(资源文件)和`AndroidManifest.xml`(应用配置文件)等。 ...

    android gridview adapter 实例

    在这个"android gridview adapter 实例"中,我们将深入探讨如何使用Adapter来填充GridView,并实现每个GridView的item(单元格)显示不同的数据。 首先,理解Adapter在Android中的作用至关重要。Adapter是连接数据...

    Android Adapter的使用

    本文将深入探讨Android适配器的使用,特别是针对初学者,旨在帮助他们更好地理解和掌握这一核心概念。 首先,我们来理解适配器(Adapter)的基本概念。适配器是Android系统中用于将数据集合(如ArrayList)映射到...

    android UI进阶之实现listview的分页加载

    本篇文章将深入探讨如何在Android中实现ListView的分页加载,提升用户体验并优化资源使用。 首先,理解分页加载的基本概念。分页加载是指在用户滚动到列表底部时,动态地从服务器或本地数据库加载更多数据,而不是...

Global site tag (gtag.js) - Google Analytics