RecyclerView 瀑布流實現(xiàn)遇到的問題及解決

說起來也是一個很久之前的問題了。那個時候還是在 17 年吧弥雹,六月份就要畢業(yè)转捕,在做面試的項目房揭,其中有一個功能用到了瀑布流來顯示妹子圖。那時候就遇到了一些問題,但是那時候確實什么都不懂矾瑰,搞了很久,有些問題還是沒有解決受扳。這個月到了月底咏闪,臨近放假了,也沒有什么事情赴涵,就開始倒騰以前的項目媒怯,發(fā)現(xiàn)了這個問題。其中也是花了一點時間髓窜,踩了一些坑扇苞。在此記錄一下欺殿。

主要的思路

主要的思路就是通過 RecyclerView 來做,通過設置 StaggeredGridLayoutManagerlayoutManager 來實現(xiàn)瀑布流的效果鳖敷。

遇到的問題

問題就是脖苏,列表圖片加載的時候會閃爍,跳動定踱,換位這幾個問題棍潘。滑動之后也同樣會出現(xiàn)這幾個問題崖媚。那時候確實什么都不懂亦歉,就上網(wǎng)查,把各種方法都試一遍畅哑,還是不行肴楷。也分析不出問題,或者說還沒有分析這個問題的能力荠呐。

其中踩到的坑

瀑布流的問題赛蔫,在網(wǎng)上還是挺多資料的。

說的最多的就是 layoutManager.setGapStrategy(StaggeredGridLayoutManager.GAP_HANDLING_NONE); 這個方法泥张,之前也不懂呵恢,反正大家都說這個可以解決(當然也有很多人說這個沒什么作用),現(xiàn)在翻了一下源碼的注釋媚创,大概了解了渗钉,就是因為瀑布流布局的特殊性,可能會有一些間隙筝野,可以通過這個 api 設置 item 間隔的處理方式晌姚,有兩個參數(shù)可以選擇 GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANSGAP_HANDLING_NONE

GAP_HANDLING_MOVE_ITEMS_BETWEEN_SPANS:當滑動空閑時歇竟,也就是 SCROLL_STATE_IDLE 這個狀態(tài)下時挥唠,會去檢查 item 之間有沒有間隙,如果有就調整 item 的位置焕议,并使用動畫的過度效果宝磨。

GAP_HANDLING_NONE:不去處理這個間隙。

設置了這個 layoutManager.setGapStrategy(StaggeredGridLayoutManager.GAP_HANDLING_NONE); 之后盅安,確實不會跳動了唤锉,但是帶來了另一個問題,就是既然不調整 item 的位置去填補這個空隙别瞭,就會有空隙窿祥,而這個空隙有多大呢?看圖:

image
image

沒錯蝙寨,看到的就是這個樣子晒衩,整個布局就亂七八糟的了嗤瞎。

網(wǎng)上的大神很多,此記不成听系,再生一記贝奇!

layoutManager.invalidateSpanAssignments()StaggeredGridLayoutManager 是會保持 item 和 span(其實我不是很理解這個東西要怎么理解) 的映射,如果要刷新這個映射就可以調用這個方法靠胜。

這個方法在什么時候調用呢掉瞳?在滑動的時候。

recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
    @Override
    public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
        super.onScrollStateChanged(recyclerView, newState);
        //刷新
        layoutManager.invalidateSpanAssignments();
    }
});

但是單純調這個方法浪漠,沒什么卵用陕习,而且布局會變得更亂,所以這個方法是在 上面layoutManager.setGapStrategy(StaggeredGridLayoutManager.GAP_HANDLING_NONE)的基礎上郑藏,在使用這個東西衡查。

雖然瘩欺,跳動的情況好了一些必盖,還是不理想,而且還會出現(xiàn)另一個問題俱饿,就是滑動到頂部會發(fā)現(xiàn)歌粥,頂部的 item 會有偏移的情況,然后拉到頂拍埠,就會刷新一下失驶,調換位置了。枣购。嬉探。。棉圈。

image

問題的根源

其實問題的根源就是圖片的高度不固定涩堤,這不廢話嘛,瀑布流的高度本來就不是固定的分瘾。胎围。。

或者換種說法德召,你聽起來或者就更明白一些了白魂。就是圖片的高度在 item 填充到 RecyclerView 的時候高度還是不確定,或者說是 0 上岗。然后圖片加載完成后福荸,高度就出現(xiàn)了,就出現(xiàn)了各種問題肴掷。

是的敬锐,只要我們解決了這個高度的問題辞嗡,其他問題就沒有了。就不需要設置前面說到的兩個設置滞造。

這個高度要怎么確定呢续室?肯定不能寫死,要根據(jù)圖片的比例來谒养,因為圖片的寬度是確定的挺狰,我們只要根據(jù)圖片的比例進行縮放就可以了。

具體的步驟是獲取到圖片买窟,然后根據(jù)測量出寬高比丰泊,再根據(jù) item 的寬度,設置 item 的高度始绍。這就行了呀瞳购!

說是很簡單啦。我之前也就想到過這個問題亏推,然后就在 Glide 里面進行攔截学赛,然后設置寬高,發(fā)現(xiàn)問題沒有解決吞杭,就沒有去細想了盏浇。其實那時候的思路大致是對的,但是那個攔截其實異步的芽狗,并沒有同步執(zhí)行绢掰,也就是說,在 Adapter 里面返回 view 的時候高度還是沒有確定童擎。所以這個問題也沒有得到解決滴劲。

解決方案

再說一下網(wǎng)絡上的解決方案是什么。

有一篇文章是這么說的顾复,其實都大致類似班挖。網(wǎng)絡上的東西,你抄我的捕透,我抄你的聪姿。獲取到圖片列表后,然后啟動一個 IntentService 來下載這一頁的圖片乙嘀,等下載完成后測量出寬高比末购,再通過 EventBus 發(fā)消息給 RecyclerView 添加數(shù)據(jù),然后 Adapter 再進行刷新虎谢。

文章里面也說了盟榴,這樣有個很大的問題就是加載出圖片的時候巨慢,因為要等這頁的圖片下載完成和測量完成婴噩,然后才進行的渲染擎场,所以速度很感人羽德。

最終的結果就是讓后臺把圖片的寬高傳過來,然后獲取到這頁數(shù)據(jù)的時候迅办,順便把寬高給設置進去宅静。

這幾乎是網(wǎng)上所有的解決方案中最終的結果(我看到的都是這樣)。

但是我調用的這個是第三方開放的 API 接口站欺,我找誰給我加數(shù)據(jù)姨夹。。矾策。磷账。

我想了一下,能不能加載一張贾虽,然后顯示一張呢逃糟?

可以的!

我要開始干了蓬豁! 思路是這樣的绰咽,我也不用 IntentService (好像網(wǎng)上還有人以為懂個 IntenService ,就洋洋得意庆尘。剃诅。)巷送,也不用 EventBus 驶忌, 用 RxJava 直接懟,因為前面這兩個東西就是 異步 和 事件 嘛笑跛,這兩個東西付魔,RxJava 都有啊。搞什么花里胡哨的飞蹂。

首先通過 Retrofit 發(fā)送請求几苍,獲取 Observer ,然后進行轉化 map 獲取到這一頁的圖片信息陈哑,再使用 flatMap 進行轉化妻坝,把這個 list 的數(shù)據(jù)逐個發(fā)送,然后再進行下載圖片惊窖,保存圖片刽宪,獲取寬高等操作,最后在 onNext 中獲取一條數(shù)據(jù)就刷新這個 item (要注意不要刷新整個列表界酒,其實前面說的會閃的問題圣拄,是因為刷新的時候,對整個列表都進行了刷新導致的)毁欣。圖片就會一個個加載出來了庇谆。不用等所有的操作都執(zhí)行完再渲染岳掐。代碼如下:

ApiRequest.getServer().getFuliData(num, page)
                // 切換線程
                .compose(ApiRequest.<BaseGankBean<FuliBean>>preRequest())
                .map(new Function<BaseGankBean<FuliBean>, List<FuliBean>>() {
                    @Override
                    public List<FuliBean> apply(BaseGankBean<FuliBean> fuliBeanBaseGankBean) throws Exception {
                        return fuliBeanBaseGankBean.getResults();
                    }
                })
                .flatMap(new Function<List<FuliBean>, ObservableSource<FuliBean>>() {
                    @Override
                    public ObservableSource<FuliBean> apply(List<FuliBean> fuliBeans) throws Exception {
                        return Observable.fromIterable(fuliBeans);
                    }
                })
                // 切換線程
                .observeOn(Schedulers.io())
                .map(new Function<FuliBean, FuliBean>() {
                    @Override
                    public FuliBean apply(FuliBean fuliBean) throws Exception {
                        String url = fuliBean.getUrl();
                        if (url!=null) {
                            try {
                                // 同步下載圖片,可設置獲取到的 bitmap 大小饭耳,但是實際下載的還是原圖
                                Bitmap bitmap = Glide.with(getActivity())
                                        .asBitmap()
                                        .load(url)
                                        .into(400, 400)
                                        .get();
                                // 保存圖片串述,作為緩存
                                ImgLoadUtil.wirteBitmap2File(bitmap, Const.FilePath.imgCache,fuliBean.get_id()+".png");
                                // 獲取寬高
                                int height = bitmap.getHeight();
                                int width = bitmap.getWidth();
                                fuliBean.setHeight(height);
                                fuliBean.setWidth(width);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            } catch (ExecutionException e) {
                                e.printStackTrace();
                            }
                        }
                        return fuliBean;
                    }
                })
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Observer<FuliBean>() {
                    @Override
                    public void onSubscribe(Disposable d) {
                        
                    }

                    @Override
                    public void onNext(FuliBean fuliBean) {
                        // 一個圖片加載完成就渲染這個圖片
                        dataList.add(fuliBean);
                        int index = dataList.indexOf(fuliBean);
                        adapter.notifyItemChanged(index);
                    }

                    @Override
                    public void onError(Throwable e) {
                        
                        e.printStackTrace();
                    }

                    @Override
                    public void onComplete() {
                        //加載完成,頁碼加一
                        page++;
                    }
                });

看完有沒有一種暢快淋漓的感覺寞肖。越發(fā)覺得 RxJava 還是很好用的剖煌。然后 Adapter 也沒什么好說的,獲取到寬高比逝淹,計算一下寬高耕姊,該怎么搞就怎么搞。

貼一下效果圖:

image
image
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末栅葡,一起剝皮案震驚了整個濱河市茉兰,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌欣簇,老刑警劉巖规脸,帶你破解...
    沈念sama閱讀 211,042評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異熊咽,居然都是意外死亡莫鸭,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評論 2 384
  • 文/潘曉璐 我一進店門横殴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來被因,“玉大人,你說我怎么就攤上這事衫仑±嬗耄” “怎么了?”我有些...
    開封第一講書人閱讀 156,674評論 0 345
  • 文/不壞的土叔 我叫張陵文狱,是天一觀的道長粥鞋。 經(jīng)常有香客問我,道長瞄崇,這世上最難降的妖魔是什么呻粹? 我笑而不...
    開封第一講書人閱讀 56,340評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮苏研,結果婚禮上等浊,老公的妹妹穿的比我還像新娘。我一直安慰自己楣富,他們只是感情好凿掂,可當我...
    茶點故事閱讀 65,404評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般庄萎。 火紅的嫁衣襯著肌膚如雪踪少。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,749評論 1 289
  • 那天糠涛,我揣著相機與錄音援奢,去河邊找鬼。 笑死忍捡,一個胖子當著我的面吹牛集漾,可吹牛的內容都是我干的。 我是一名探鬼主播砸脊,決...
    沈念sama閱讀 38,902評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼具篇,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了凌埂?” 一聲冷哼從身側響起驱显,我...
    開封第一講書人閱讀 37,662評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎瞳抓,沒想到半個月后埃疫,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,110評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡孩哑,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年栓霜,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片横蜒。...
    茶點故事閱讀 38,577評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡胳蛮,死狀恐怖,靈堂內的尸體忽然破棺而出愁铺,到底是詐尸還是另有隱情鹰霍,我是刑警寧澤,帶...
    沈念sama閱讀 34,258評論 4 328
  • 正文 年R本政府宣布茵乱,位于F島的核電站,受9級特大地震影響孟岛,放射性物質發(fā)生泄漏瓶竭。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,848評論 3 312
  • 文/蒙蒙 一渠羞、第九天 我趴在偏房一處隱蔽的房頂上張望斤贰。 院中可真熱鬧,春花似錦次询、人聲如沸荧恍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽送巡。三九已至摹菠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間骗爆,已是汗流浹背次氨。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留摘投,地道東北人煮寡。 一個月前我還...
    沈念sama閱讀 46,271評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像犀呼,于是被迫代替她去往敵國和親幸撕。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,452評論 2 348

推薦閱讀更多精彩內容