摘要:前言安卓開發(fā)者都知道,比要靈活的多,但不可否認的里面的坑也同樣埋了不少人。下面讓我們看看騰訊開發(fā)工程師用實例講解自己踩坑時的解決方案和心路歷程。
前言
安卓開發(fā)者都知道,RecyclerView比ListView要靈活的多,但不可否認的里面的坑也同樣埋了不少人。下面讓我們看看騰訊開發(fā)工程師用實例講解自己踩坑時的解決方案和心路歷程。
話說有圖有真相,首先來對比一下局部刷新前后的效果:
優(yōu)化之前的效果:
優(yōu)化之后的效果:
可以看到,優(yōu)化之后,列表中的這張大圖不在有一閃一閃亮晶晶的效果了!
那么,這是如何做到的呢?這是本文的重點,本文的大綱主要包括:
?一、為什么會閃一下?分析為什么會閃一下
對分析的可能造成閃動的問題進行解決
驗證是否解決
我們的需求是大家已經看到了,點擊打分,彈出一個對話框,點擊一個分數(shù),這時候,通過一些列復雜的轉換(當然不是本文的論述的重點),這時候到了要更新列表項了,如是很自然,我們會這么做:
因為,操作的那個列表項你是知道他的position,所以你可以這么做,(當然,我之前是直接notifyDataSetChanged的,這個會照成所以不不要的item也會刷新)然而,閃動還是出現(xiàn)了,那么我開始懷疑:
?二、帶著思考,就去嘗試吧!流傳甚為廣泛的一種說法,imageView的寬高不固定導致的(wrap_content)?
這個是RecyclerView自帶的更新動畫效果導致的?
這個是因為圖片加載框架(glide 的 animte)的動畫效果導致的?
getView中(RecyclerView中是onBindViewHolder)加載圖片的時候,設置一個tag,當發(fā)現(xiàn)這個imageView的tag和之前的tag一致時就不加載
1、對于第一種,我的做法是自己寫了一個自定義的imageView,重寫omMeasure方法,如下:
因為我們的這個列表項中的圖片是(高=寬)的,因此,我才這么寫,這樣寫也有一個好處,不用在onBindViewHolder中去動態(tài)的計算出高度,然后在已layoutParm的方式設置給imageView,相信不少小伙伴都做過了吧!
然而,遺憾的是,他并沒有解決閃一下的問題!此時這個閃動的原因顯然不在這里,但是這里做的,可以保留下來。
2、對于第二種說法,我參考了這里http://stackoverflow.com/ques...
的做法:
以及也嘗試了這種
然而,那種漸變的閃動消失了,但是,取而代之的是一種更加不可接受的閃動,這里就不用gif展示了,因此原因也并不在此處。
3、對于對三種說法,我也去嘗試了一下將glide加載改為:
然而得到的依然是一個失望的結果,依然沒有解決閃動的問題,原因也不在此處。
4、那么,就剩下最后一個猜測了,那么會不會是它呢?那就試試吧,于是代碼改為:
這里的做法其實就是設置Tag,那么是騾子是馬,拉出來溜溜吧,結果更加令人發(fā)指,如圖:
好吧,此時已經有點崩潰了,顯然這個也不是我要的結果,那么此時是否應該在靜下來想一想,自己對于可能的幾種原因做過的一些對策,是否有哪里遺漏了。經過思考,發(fā)現(xiàn)并沒有?。∧敲匆欢ㄊ沁€有其他的原因,沒有考慮到!
還是去翻一翻RecyclerView的api吧,我注意到了這個api:
可以看到這里有一個payload的參數(shù),use null to identify a "full" update這是說如果傳null就是全部更新,回過頭去看一看我們之前的調用方式:
看一下源碼,發(fā)現(xiàn)
實際上,payload這個參數(shù)就是傳的null,那也就是說如果傳一個不為null的參數(shù),就可以對列表項中的具體控件更新了?
http://stackoverflow.com/ques...
我了解到這個方法的使用方式是這樣的:
然來,onBindViewHolder有這么一個重載方式,如是我也這么做了,在下面這個重載中,去更新我想更新的控件:
然后,更新的方式變成了這種:
是騾子是馬,那就在遛一遛吧!
然而,依然是會閃一下!??!這這么會!??!還是調試一下吧,新重載onBindViewHolder方法有沒有被執(zhí)行,一更代碼,發(fā)現(xiàn)果然沒有被執(zhí)行!那么,究竟是什么鬼?去網上查了一下,有人給出了一個解決辦法:
http://stackoverflow.com/ques...
需要重寫這個動畫,讓永遠返回true,已達到newHolder和olderHolder是同一個,然而,這真的就是我的救命稻草嗎?
那么,是騾子是馬,拉出來溜溜吧,然而,并不是馬!!進源碼看一看
發(fā)現(xiàn)其實只要我們傳入的payload不為空,那么返回的就是true?重寫有意義嗎?顯然,我重載的onBindViewHolder方法并沒有執(zhí)行的原因顯然不是這個。
那么,到底,到底問題出在何處?會不會是XrecyclerView的問題?根據調用棧,我看到第一個onBindViewHolder被執(zhí)行了,往上面跟,發(fā)現(xiàn)XrecyclerView的實現(xiàn)果然存在問題!
如圖,作者僅僅只實現(xiàn)了,不帶payload的方法,最后adapter調用的只有不帶paylaod的方法!所以,重寫一個吧!
最后!終于達到了想要的效果了,經過這次爬坑,選擇一個開源的框架真滴是需要慎重再慎重。
總結實際上RecyclerView做局部刷新是非常容易的,其實就是使用好帶payload參數(shù)的這個notifyItemRangeChanged方法,以及override帶payload的這個onBindViewHolder方法,在onBindViewHolder中去刷新你想更新的控件即可,并非是網上傳聞的那些原因,當然此處爬坑時間之長,也可能更選用開源控件不當有關,所以,選擇開源控件,要謹慎再謹慎!
原文鏈接:http://wetest.qq.com/lab/view...
閱讀更多
kotlin學習筆記-異常好玩的list集合總結
一份完整的Android Studio搭建Flutter教程
NDK項目實戰(zhàn)—高仿360手機助手之卸載監(jiān)聽
(Android)面試題級答案(精選版)
相信自己,沒有做不到的,只有想不到的在這里獲得的不僅僅是技術!
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://www.ezyhdfw.cn/yun/71634.html
閱讀 2771·2021-09-28 09:35
閱讀 3407·2021-09-03 10:28
閱讀 3055·2019-08-30 15:43
閱讀 1610·2019-08-30 14:04
閱讀 1966·2019-08-29 17:02
閱讀 1964·2019-08-26 13:59
閱讀 936·2019-08-26 11:51
閱讀 3436·2019-08-23 17:16