赞
踩
12年毕业,专科生,做安卓9年了。横扫深圳各大互联网公司,从开始的4k工资到现在的4万。我经历了什么?
一类:腾讯,支付宝,头条,快手,OPPO,VIVO, 百度,华为二类A组:北京:滴滴,京东, 美团,爱奇艺,小米,拼多多,360,新浪,携程,微博,哔哩哔哩 ,搜狐,知乎,联想,旷视,优酷,58同城,贝壳找房,作业帮,创新工场,金山深圳:大疆,顺丰,恒大,万科,货拉拉,TCL,平安,荣耀, 美的,Shopee,一加,随手记,中兴广杭:网易 ,海康威视,斗鱼,小红书,去哪儿,喜马拉雅,创维,虎牙,涂鸦智能,51信用卡,猿辅导二类B组:海能达,金蝶,有赞,富涂,深信服,广和通, 移卡,珍爱网,步步高,优必选科技,柔宇科技,迅雷,芒果网,吉比特,传音控股唯品会,猎豹,科大讯飞,格力
IdleHandler 是 MessageQueue 的静态内部接口。
IdleHandler,这是一种在只有当消息队列没有消息时或者是队列中的消息还没有到执行时间时才会执行的 IdleHandler,它存放在mPendingIdleHandlers队列中。
平时是如何使用的:
Looper.myQueue().addIdleHandler(new MessageQueue.IdleHandler() {
@Override
public boolean queueIdle(){
//空闲时处理逻辑
return false;
}
});
从之前的类定义可以看出,返回 false 表示执行后就将该回调移除掉,返回 true 表示该 IdleHandler 一直处于活跃状态,只要空闲就会被回调。
1)首先说下RecycleView的缓存结构:
Recycleview有四级缓存,分别是mAttachedScrap(屏幕内),mCacheViews(屏幕外),mViewCacheExtension(自定义缓存),mRecyclerPool(缓存池)
mAttachedScrap(屏幕内)
,用于屏幕内itemview快速重用,不需要重新createView和bindViewmCacheViews(屏幕外)
,保存最近移出屏幕的ViewHolder,包含数据和position信息,复用时必须是相同位置的ViewHolder才能复用,应用场景在那些需要来回滑动的列表中,当往回滑动时,能直接复用ViewHolder数据,不需要重新bindView。mViewCacheExtension(自定义缓存)
,不直接使用,需要用户自定义实现,默认不实现。mRecyclerPool(缓存池)
,当cacheView满了后或者adapter被更换,将cacheView中移出的ViewHolder放到Pool中,放之前会把ViewHolder数据清除掉,所以复用时需要重新bindView。2)四级缓存按照顺序需要依次读取。所以完整缓存流程是:
itemView
时,先把屏幕内的ViewHolder保存至AttachedScrap
中CacheView
,CacheView大小默认是2,超过数量的话按照先入先出原则,移出头部的itemview保存到RecyclerPool缓存池
(如果有自定义缓存就会保存到自定义缓存里),RecyclerPool缓存池会按照itemview的itemtype
进行保存,每个itemType缓存个数为5个,超过就会被回收。CacheView
中获取,也是通过pos获取holder缓存自定义缓存
中获取缓存——>获取失败,从mRecyclerPool
中获取viewholder
——createViewHolder并bindview。3)了解了缓存结构和缓存流程,我们再来看看具体的问题
滑动10个,再滑回去,会有几个执行onBindView?
onBindView
的只有一种缓存区,就是缓存池mRecyclerPool
。所以我们假设从加载RecyclView
开始盘的话(页面假设可以容纳7条数据):
onCreateViewHolder
和onBindViewHolder
。mCacheViews
中。此时mCacheViews
缓存区数量为1,mRecyclerPool
数量为0。然后新出现的position=7的数据通过postion在mCacheViews
中找不到对应的ViewHolder
,通过itemtype
也在mRecyclerPool
中找不到对应的数据,所以会调用onCreateViewHolder
和onBindViewHolder
方法。mCacheViews
中,但是由于mCacheViews
缓存区默认容量为2,所以position=0的数据会被清空数据然后放到mRecyclerPool
缓存池中。而新出现的position=9数据由于在mRecyclerPool
中还是找不到相应type的ViewHolder,所以还是会走onCreateViewHolder
和onBindViewHolder
方法。所以此时mCacheViews
缓存区数量为2,mRecyclerPool
数量为1。mRecyclerPool
中找到相同viewtype的ViewHolder了。所以就直接复用了,并调用onBindViewHolder
方法绑定数据。mCacheViews
中,再出现的时候是不会调用onBindViewHolder方法,而复用的第三条数据是从mRecyclerPool
中取得,就会调用onBindViewHolder
方法了。4)所以这个问题就得出结论了(假设mCacheViews
容量为默认值2):
bindview
方法。然后滑回去,会走10-2个bindview
方法。一共18次调用。bindview
方法。然后滑回去,会走10-2个bindview
方法。一共16次调用。但是但是,实际情况又有点不一样。因为Recycleview
在v25版本引入了一个新的机制,预取机制
。
预取机制
,就是在滑动过程中,会把将要展示的一个元素提前缓存到mCachedViews
中,所以滑动10个元素的时候,第11个元素也会被创建,也就多走了一次bindview
方法。但是滑回去的时候不影响,因为就算提前取了一个缓存数据,只是把bindview
方法提前了,并不影响总的绑定item数量。
所以滑动的是新数据的情况下就会多一次调用bindview
方法。
5)总结,问题怎么答呢?
bindview
可以是19次调用,可以是16次调用。viewholder
。cachedView
就是mCacheViews
缓存区中的view,是不需要重新绑定数据的。Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。