Android RecyclerView自动翻页方案

作者:MrTrying

其实已经有很多上拉加载更多、或者滑动到底自动加载的自定义RecyclerView,这里所使用的方案是通用于RecyclerView的,目的就是为了提高代码复用

通常在app的列表中会使用分页加载数据,当用户停止滑动列表到达底部时会加载下一页数据;为了更好地用户体验,可以在列表停止滑动是会提前几个item加载下一页数据。

public static void setLoad(@NonNull RecyclerView recyclerView, final OnLoadCallback onLoadCallback) {
   recyclerView.setOnScrollListener(new RecyclerView.OnScrollListener() {
       int visibleLast = -1;

       @Override
       public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
           super.onScrollStateChanged(recyclerView, newState);
           if(newState == SCROLL_STATE_IDLE){
               RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
               if(layoutManager == null){
                   return;
               }
               if (layoutManager instanceof LinearLayoutManager) {
                   visibleLast = ((LinearLayoutManager) layoutManager).findLastVisibleItemPosition();
               } else if (layoutManager instanceof StaggeredGridLayoutManager) {
                   int[] lastItemArr = ((StaggeredGridLayoutManager) layoutManager).findLastVisibleItemPositions(null);
                   if (lastItemArr.length > 0) {
                       visibleLast = lastItemArr[lastItemArr.length - 1];
                   }
               }
               if (recyclerView.getAdapter() != null
                       && recyclerView.getAdapter().getItemCount() - 4 <= visibleLast) {
                   if (onLoadCallback != null) {
                       onLoadCallback.onLoad();
                   }
               }
           }
       }
   });
   if (onLoadCallback != null) {
       onLoadCallback.onLoad();
   }
}

这个需求代码还是比较简单的,基于这个需求,基本实现思路就是为RecyclerView设置滑动监听,在回调中处理请求下一页数据的回调。

上面的代码有个问题,实际使用中onLoadCallback可能会被重复回调

public static void setLoad(@NonNull RecyclerView recyclerView, final OnLoadCallback onLoadCallback) {
        recyclerView.setOnScrollListener(new RecyclerView.OnScrollListener() {
            int visibleLast = -1;
            int totalCount;
            boolean allow = true;

            @Override
            public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
                if(newState == SCROLL_STATE_IDLE){
                    RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
                    if(layoutManager == null){
                        return;
                    }
                    if (layoutManager instanceof LinearLayoutManager) {
                        visibleLast = ((LinearLayoutManager) layoutManager).findLastVisibleItemPosition();
                    } else if (layoutManager instanceof StaggeredGridLayoutManager) {
                        int[] lastItemArr = ((StaggeredGridLayoutManager) layoutManager).findLastVisibleItemPositions(null);
                        if (lastItemArr.length > 0) {
                            visibleLast = lastItemArr[lastItemArr.length - 1];
                        }
                    }
                    if(recyclerView.getAdapter() != null){
                        if(!allow){
                            allow = totalCount != recyclerView.getAdapter().getItemCount();
                        }
                        if (allow && recyclerView.getAdapter().getItemCount() - 4 <= visibleLast) {
                            if (onLoadCallback != null) {
                                onLoadCallback.onLoad();
                            }
                        }

                    }
                }
            }
        });
        if (onLoadCallback != null) {
            onLoadCallback.onLoad();
        }
    }

这里添加了一个allowboolean变量和totalCounttotalCount用于记录当前adapter中的数据总数,如果totalCountadapter.getItemCount()不相等,说明数据发生变化,可以再继续下一次加载。

这种逻辑方式基于请求有数据返回的基础之上,如果出现那种请一次有数据更新,再一次没有,还需要在继续请求的情况是做不到的,如果有这种比较任性的需求,还是自己控制这个是否可以继续加载的标识位比较靠谱

随后,产品又招上我了,说:“为什么这个页面自动加载的这么慢,其他页面没有问题?”。这是一个网格列表,当快速滑动都底部时,等待了2s左右的时间才刷新出来下一页的数据,这TMD就有点诡异了。

一查发现,RecyclerView快速滑动到底之后,并没有马上请求下一页的数据。RecyclerView在设置为GridLayoutMannager时,RecyclerView快速滑动后继续惯性滑动到底时,OnScrollListeneronScrollStateChanged()SCROLL_STATE_SETTLING状态变更到SCROLL_STATE_IDLE状态花了1.8~2.4s左右的时间(开始怀疑人生,难道是我打开方式不对?)。

没招了,只能将代码从onScrollStateChanged()放到onScrolled()里,这样一来确实有一部分的性能损失,但是这一情况太影响用户体验之只能这么办。适当的调整了一下代码的逻辑,尽量减少无效的findItem计算。

public static void setLoad(@NonNull RecyclerView recyclerView, final OnLoadCallback onLoadCallback, boolean isAuto) {
    recyclerView.setOnScrollListener(new RecyclerView.OnScrollListener() {
        int visibleLast = -1;
        int totalCount;
        boolean allow = true;

        @Override
        public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) {
            super.onScrolled(recyclerView, dx, dy);
            if (recyclerView.getAdapter() != null) {
                if (!allow) {
                    allow = totalCount != recyclerView.getAdapter().getItemCount();
                }
            }
            if (!allow) {
                return;
            }
            RecyclerView.LayoutManager mLayoutManager = recyclerView.getLayoutManager();
            if (mLayoutManager == null) {
                return;
            }
            if (mLayoutManager instanceof LinearLayoutManager) {
                visibleLast = ((LinearLayoutManager) mLayoutManager).findLastVisibleItemPosition();
            } else if (mLayoutManager instanceof StaggeredGridLayoutManager) {
                int[] lastItemArr = ((StaggeredGridLayoutManager) mLayoutManager).findLastVisibleItemPositions(null);
                if (lastItemArr.length > 0) {
                    visibleLast = lastItemArr[lastItemArr.length - 1];
                }
            }
            if (recyclerView.getAdapter() != null
                    && recyclerView.getAdapter().getItemCount() - 4 <= visibleLast
                    && allow) {
                allow = false;
                totalCount = recyclerView.getAdapter().getItemCount();
                if (onLoadCallback != null) {
                    onLoadCallback.onLoad();
                }
            }

        }
    });
    if (onLoadCallback != null) {
        onLoadCallback.onLoad();
    }
}

如果有大佬有更好的解决方案,还请指点

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,888评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,705评论 2 59
  • 朋友圈里新春的欢乐气氛还没有退去,就被各种上映的开学季大片所淹没。班级群里扑面而来老师们的各种学习习惯要求,而妈妈...
    小鱼艾小米阅读 198评论 0 1
  • 週末晚上,回家陪老媽老爸看“星光大道”,一不留神被戳中淚點。 節目裏的一個嘉賓叫“毛毛”,是一個不...
    欢乐V英雄阅读 231评论 0 3
  • 21天微演讲打卡练习第5天,每天1分钟,成为最美好的自己。 当自己从一个嗷嗷待哺的婴儿成长到今天,经历了许许多多的...
    吴彩凤1阅读 199评论 0 1