Android 面试总结之ListView

本文详细介绍了ListView的高效使用技巧,包括数据集更新、分页加载、多种类型Item显示、定位、监听器使用、图片加载优化、多线程更新、性能提升策略及替代方案。深入解析ListView原理,并推荐了几个实用的开源ListView控件。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、当 ListView 数据集改变后,如何更新 ListView

使用该 ListView 的 adapter 的 notifyDataSetChanged()方法。该方法会 使 ListView 重新绘制。

2、ListView 如何实现分页加载

设 置 ListView 的 滚 动 监 听 器 : setOnScrollListener(new OnScrollListener{….}) 
在监听器中有两个方法: 滚动状态发生变化的方法 (onScrollStateChanged)和 listView 被滚动时调用的方法(onScroll)
在滚动状态发生改变的方法中,有三种状态: 
手指按下移动的状态: SCROLL_STATE_TOUCH_SCROLL: // 触摸滑动 
惯性滚动(滑翔(flgin)状态): SCROLL_STATE_FLING: // 滑翔 静止状态 
SCROLL_STATE_IDLE: // 静止
对不同的状态进行处理: 
分批加载数据,只关心静止状态:关心最后一个可见的条目,如果最后一个 可见条目就是数据适配器(集合)里的最后一个,此时可加载更多的数据。在每 次加载的时候,计算出滚动的数量,当滚动的数量大于等于总数量的时候,可以 提示用户无更多数据了。

3、ListView 如何显示多种类型的Item

ListView 显示的每个条目都是通过 baseAdapter 的 getView(int position, View convertView, ViewGroup parent)来展示的,理 论上我们完全可以让每个条目都是不同类型的 view。 比如:从服务器拿回一个标识为 id=1,那么当 id=1 的时候,我们就加载类 型一的条目,当 id=2 的时候,加载类型二的条目。常见布局在资讯类客户端中 可以经常看到。 
除此之外 adapter 还提供了 getViewTypeCount()和 getItemViewType(int position)两个方法。在 getView 方法中我们可以根据不 同的 viewtype 加载不同的布局文件。
 

4、ListView 如何定位到指定位置? 


可以通过 ListView 提供的 lv.setSelection(listView.getPosition());方法。

5、在ListView中设置Selector为null会报空指针? 


mListView.setSelector(null);//空指针 
试试下面这个: 
mListView.setSelector(new ColorDrawable(Color.TRANSPARENT));

6、ListView中主要有2种监听器:

 
OnItemClickListener: 处理视图中单个条目的点击事件。 (OnItemSelectedListener,参数与其一致,只是监听用户手指行为不同) 
OnScrollListener:监听滚动变化,可以用于视图滚动中加载数据。

7、ListView中图片错位问题 


每次getView能给对象一个标识,在异步加载完成时比较标识与当前行item的标识是否一致,一致则显示,否则不做处理即可。
 

//给ImageView设置一个Tag
holder.img.setTag(imgUrl);
//预设一张图片
holder.img.setImageResource(R.drawable.ic_launcher);

//防止图片错位
if(imageView.getTag() != null && imageView.getTag().equals(imageUrl)){
    imageView.setImageBitmap(result);
}

8、ListView中多线程更新问题 


使用Handler负责统一刷新, ListView带有进度条的每个Item,多个子线程刷新各自的进度,如果子线程很多那么刷新就会变得很频繁,我们可以由一个Handler负责统一刷新,这样我们就要以增加一些额外条件限制刷新的次数和条件。

9、ListView 如何提高其效率? 


a. 当 convertView 为空时,用 setTag()方法为每个 View 绑定一个存放控件的 ViewHolder 对象。当 convertView 不为空,重复利用已经创建的 view 的时候, 使用 getTag()方法获取绑定的 ViewHolder 对象,这样就避免了 findViewById 对控件的层层查询,而是快速定位到控件。

b. 复用 ConvertView,使用历史的 view,提升效率 200%,自定义静态类 ViewHolder,减少 findViewById 的次数。提升效率 50%。
c. 异步加载数据,分页加载数据。
d. 使用 WeakRefrence 引用 ImageView 对象。

 

10、ListView中加载优化图片图片的优化策略

1、处理图片的方式: 
如果 ListView 中自定义的 Item 中有涉及到大量图片的,一定要对图片进行 细心的处理,因为图片占的内存是 ListView 项中最头疼的,处理图片的方法大 致有以下几种:

a. 不要直接拿路径就去循环 BitmapFactory.decodeFile

b. 使用 Options 保存图片大小,不要加载图片到内存去
c. 对图片一定要经过边界压缩尤其是比较大的图片,如果你的图片是后台 服务器处理好的那就不需要了
d. 在 ListView 中取图片时也不要直接拿个路径去取图片,而是以 WeakReference(使用 WeakReference 代替强引用。比如可以使用 WeakReference mContextRef、SoftReference、WeakHashMap 等的来存 储图片信息。
e. 在 getView 中做图片转换时,产生的中间变量一定及时释放


2、异步加载图片基本思想: 
1). 先从内存缓存中获取图片显示(内存缓冲) 
2). 获取不到的话从 SD 卡里获取(SD卡缓冲) 
3). 都获取不到的话从网络下载图片并保存到 SD 卡同时加入内存并显示(视情况看是否要显示)

注:优化基本原理

优化一: 先从内存中加载,没有则开启线程从 SD 卡或网络中获取,这里注 意从 SD 卡获取图片是放在子线程里执行的,否则快速滑屏的话会不够流畅。

优化二: 于此同时,在 adapter 里有个 busy 变量,表示 listview 是否处于 滑动状态,如果是滑动状态则仅从内存中获取图片,没有的话无需再开启线程去 外存或网络获取图片。

优化三: ImageLoader 里的线程使用了线程池,从而避免了过多线程频繁 创建和销毁,如果每次总是 new 一个线程去执行这是非常不可取的,好一点的 用的 AsyncTask 类,其实内部也是用到了线程池。在从网络获取图片时,先是 将其保存到 sd 卡,然后再加载到内存,这么做的好处是在加载到内存时可以做 个压缩处理,以减少图片所占内存。
 

11、ListView实现Item局部刷新?

private void updateView(int itemIndex){
    //得到第一个可显示控件的位置
    int visiblePosition = mListView.getFirstVisiblePosition();
    //如果当腰更新的view在可见的位置时就更新,否则跳过
    if(itemIndex - visiblePosition  >= 0){
        //得到要更新的item的view
        View view = mListView.getChidAt(itemIndex - visiblePosition);
        //调用adapter更新界面
        mAdapter.updateView(view,itemIndex);
    }
}

12、Listview 的替代方案

RecyclerView是一个比ListView更灵活的一个控件,以后可以直接抛弃ListView了,也可以替代Gridview。 RecyclerView改善了ListView的编程接口,他其实是一个ViewGroup ,能配合任何基于adapter的view实现多种布局。更好的实现局部刷新 ,调用notifyItemChanged(position)即可。

13、Listview 原理(重要)


首先要理解Listview 三个重要的成员,用MVC思路来理解: 
* ListVeiw: 用来展示列表的View(V) 
* Adapter : 用来把数据映射到ListView上(C) 
* 数据:item显示的数据(M)

ListView 针对每个item,要求 adapter “返回一个视图” (getView),也就是说ListView在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得到ListView的长度,然后根据这个长度,调用getView()一行一行的绘制ListView的每一项。如果你的getCount()返回值是0的话,列表一行都不会显示,如果返回1,就只显示一行.如果我们有几千几万甚至更多的item要显示怎么办?为每个Item创建一个新的View?不可能!实际上Android早已经缓存了这些视图.

可以通过下图来理解:

当要显示一个View就调用一次这个方法。这个方法是ListView性能好坏的关键。方法中有个convertView,这个是Android在为我们而做的缓存机制。 ListView中每个item都是通过getView返回并显示的,假如item有很多个,那么重复创建这么多对象来显示显然是不合理。因此Android提供了Recycler,将没有正在显示的item放进RecycleBin,然后在显示新视图时从RecycleBin中复用这个 View。

Recycler可以理解为一个回收池,存放着不同ViewType的convertView。 Recycler的工作原理大致如下:

假设屏幕最多能看到11个item,那么当第1个item滚出屏幕,这个item的View进入RecycleBin中,第12个要出现前,通过 getView从回收站(RecycleBin)中重用这个View,然后设置数据,而不必重新创建一个View。

14、有哪些开源的ListView控件?

android-pulltorefresh :
一个强大的拉动刷新开源项目,支持各种控件下拉刷新,ListView、ViewPager、WebView、ExpandableListView、GridView、ScrollView、Horizontal ScrollView、Fragment 上下左右拉动刷新,比下面 johannilsson 那个只支持 ListView 的强大的多。并且它实现的下拉刷新 ListView 在 item 不足一屏情况下也不会显示刷新提示,体验更好。 
Github:https://github.com/chrisbanes/Android-PullToRefresh

Android-PullToRefreshRecyclerView :
支持下拉刷新的RecyclerView,同时支持滑动到底部自动加载数据、给RecyclerView添加Header。并且不更改原有RecyclerView的逻辑。 
Github:https://github.com/HomHomLin/Android-PullToRefreshRecyclerView

android-pulltorefresh-listview :
下拉刷新 ListView,这个被很多人使用的项目实际有不少 bug 
Github:https://github.com/johannilsson/android-pulltorefresh

SwipeListView :
支持定义 ListView 左右滑动事件,支持左右滑动位移,支持定义动画时间 
Github:https://github.com/47deg/android-swipelistview

pinned-section-listview: 
GroupName 滑动到顶端时会固定不动直到另外一个 GroupName 到达顶端的 ExpandListView 
Github:https://github.com/beworker/pinned-section-listview
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值