美文网首页AndroidAndroidUI效果仿写
ScrollView中Recyclerview显示不全

ScrollView中Recyclerview显示不全

作者: moushao | 来源:发表于2017-08-04 10:23 被阅读6452次

    背景

    公司项目开发,首页有新闻、banner等其他东西,一页肯定显示不全,所以套了个ScrollView。新闻一共只显示4条,默认全部显示,滑动的时候只滑动ScrollView,!当然,新闻需要用到Ryc。首页做完都快两个月了,昨日猛然惊(我)喜(艹)发现,在安卓6.0的机子上,无论后台返回的新闻条数是多少(大于2,正常情况下只返回4条),界面上都只显示2条新闻,闹鬼了!

    仔细看底部导航栏的投影,我是拉到底的.gif

    分析

    这种问题,首先想到的是数据设置出错,打断点调试!没错!那我想应该是Adapter或者ViewHolder的错,调试发现,Adapter中,虽然getCount()获取到的数量是正确的,但是,onCreateView和OnBaindView的方法始终只调用了两次,没有全部调完!
    网上氹了许久,皆不知为何,遂自我分析。

    首先,自打用Rcyc起,便一直钟爱这万能BaseRecycAdpter,至今一年半载从未出过差错,其他界面的rcyc也是正常显示,故可基本排除,非此引起!
    其次,此界面亦用了ScollerView,两者属性特点皆为滑动,代码中,设置了Scoller下拉到顶部就触发刷新数据,两者滑动是否存在冲突?

    实验

    既然如此,重新创建一个TestActivity,layout布局文件先使用自动生成的,把首页布局中的RcyclerView拷贝进去,同样的数据同样的初始化,运行!能够正常显示,没一点问题!!!然后,让TextActivity加首页的布局,其他什么都不变,运行!能显示三个,但不能完全显示。我再gone了首页所有其他的组件,只留下rcyc,运行,漂亮,在首页中,rcyc显示正常!!!

    结论

    样本的唯一性已经表明,首页rcyc显示不完全的问题,是和Scrollerview的嵌套产生的,rcyc的核心是ViewHolder,重复利用视图,rcyc之所以显示两个有两个原因:
    1 rcyc在Scrollerview中重写高度失败,只能显示两个item的高度
    2 重写了LinerLayoutManager的canScrollVertically()方法,禁止了Rcyc的垂直滑动

    两则导致的结果,直接让人产生的错觉是rcyc只显示两个。如果不重写canScrollVertically()方法,可以滑动看到其他item,但是rcyc的高度依然只有两个item的高度。

    解决办法:

    网上查询ScrollView嵌套RecyclerView的问题,千篇一律说的都是重写LinearLayoutManager,试了几次,无爱心累,解决不了我的问题!

    再次分析,是不是scrollview没有下拉到底,所以rcyc显示不完全,在rcyc后新加一个可视化控件,运行,能显示新加的控件,说明scroller能够下拉到底,但rcyc还是只显示两个item的高度。

    或许是疾病乱投医,疯了!!!我直接给Rcyc外嵌套一个布局,再设置 android:descendantFocusability="blocksDescendants"
    让它直接获得焦点(这个属性可以不用加):

        <RelativeLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:descendantFocusability="blocksDescendants">
                <android.support.v7.widget.RecyclerView
                    android:id="@+id/infor_recycle"
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginLeft="10dp"
                    android:layout_marginRight="10dp"
                    android:layout_marginTop="5dp"
                    android:background="#FFFFFF">
                </android.support.v7.widget.RecyclerView>
            </RelativeLayout>
    

    运行一看,哎哟我去,激动万分啊!!!终于可以了!!!!android 6.0的机子上能正常显示了。
    巴特、巴特!似乎有点小问题,scroller滑动起来好像不怎么流畅!说白了还是嵌套滑动问题,直接设置

     recyclerview.setnestedscrollingenabled(false);//禁止rcyc嵌套滑动
    

    试运行!噢啦!全部解决,老子这个鸡冻哟,搞了将近四个小时!

    true.gif

    总结

    我并没有重写LinearLayoutManager,只是嵌套了一层布局。但为什么这样就可以了,我并没弄清楚!

    碎碎念

    无论多么复杂的首页布局,其实直接用一个Rcyc就可以搞定,而且scroller中嵌套rcyc本身就是个错误,
    参考 VLayout适配器的万能封装

    更多问题加群:584275290

    相关文章

      网友评论

      • shpunishment:感谢啊!查了都说重写LinearLayout,还是不行。重新加个布局就可以,神奇!:grin:
        moushao:@小眼睛sier 现在在写之前,一般都是避开Scroller和Recyc的嵌套,常规下采用VLayout的Recyc高级应用,多布局。如果还不能满足,或者觉得麻烦,可以试试Scroll+Fragment,Fragment里来写Recyc,Fragment用FramLayout装,不要用Viewpager
        59287c99cc43:你不用这种,那请问你现在用哪种啊?复写adapter吗?好麻烦啊
        moushao:文章我已经提到了Recyc+Scorller本身就是个bug,真的不建议这种姿势,我已经完全放弃了
      • OkTry:敢问一下作者 你所说的BaseRecycAdpter 是个什么东西 你自定义的吗?我也想自己定义个通用型适配器,作者可以推荐下吗 感谢:smiley:
        moushao:对,万能适配器.在文章末尾有[VLayout适配器的万能封装](https://www.jianshu.com/p/6383b182092e),这个适配器是在传统万能适配器的基础上,结合了阿里的DelegetAdapter,就可以做多多样式布局.当然,传统的适配器直接add到阿里的DelegetAdapter也可以实现多样式布局.
        OkTry:因为无从下 手。。有没有相关资料和案例 为我指点迷津。:innocent: :yum:
      • 4f6b638e3d98:使用NestedScrollView代替ScrollView可以直接解决这个问题。
        moushao:@小蘑菇的驼羊 和BRVAH大差不差,但我没用过它,头部是几个按钮的话,参考https://www.jianshu.com/p/6383b182092e中第二个GridAdapter
        小蘑菇的驼羊:@牟仯 多样式布局是不是需要后台数据的支持?如果头部是一个本地的几个按钮,第二部分也是本地的一些图,该如何操作?这个是不是和BRVAH有点类似啊
        moushao:@xiaoqingxu0502 是可以,但我已经习惯用VLayout了!
      • 4f6b638e3d98:如何你讲targetSdkVersion修改为21,你会有惊喜。
        moushao:现在都是用阿里的VLayout了,这种惊喜还是算了,哈哈!
      • 稻草僧:将ScrollView改成NestedScrollView即可
        moushao:每个人喜欢用的都不一样,recycle的高级应用,上瘾!
        稻草僧:@左半边翅膀丶 看来英雄所见略同
        今夜相思又几许:其实我也想说这个
      • Jay_hao:原因很简单,其实就是在RelativeLayout布局里把RecyclerView里的所有item展示出来,然后屏蔽RecyclerView的滑动监听,使用ScrollView的滑动监听
        moushao:@Jay_hao 所以,如此一来,recyc就失去了意义
      • 修得养得梦得过得:老铁666,解决问题没毛病
        moushao:@修得养得梦得过得 老铁,千万千万不要用这种嵌套,后期坑死你不偿命.赶快改用recyc的高级布局
      • Pyro:请教一下,如果设置RecyclerView的高度 android:layout_height="wrap_content"
        有人说,这样设置的话,必须一次性读取所有元素,RecyclerView的回收机制就不起作用了
        moushao:@pyrotechnist 你看到微博,淘宝,支付宝很多界面,或者新闻列表界面,其实大多都是一个Recyc搞定的,你可以看看这边文章:http://www.jianshu.com/p/6383b182092e
        Pyro:@牟仯 再次请教,什么是高级布局?
        moushao:如果是嵌套,回收机制的确作废,我试过.不嵌套用wrap_content没问题.
        还是建议使用高级布局吧,用会了,真的会上瘾!
      • 顾小浪:ScrollView嵌套RecyclerView使用就是个错误,RecyclerView就行了。
        南霂风: @王简芝 @南霂风 先嵌套一个线性布局recyerview依然会显示不全或者完全不显示
        南霂风: @王简芝 小白想问下为什么ScrollView嵌套RecyclerView使用是个错误😬
        moushao:我也考虑到这个问题了,但这都是后期改动的,没办法整改!工程量太大了!
      • 墨源:你需要检查RecyclerView的onCreateViewHolder()方法是怎么写的,如果不玩花的,RecyclerView无论如何嵌套都是可以显示所有条目的。
        moushao:Rcyc的核心是Viewholder机制,假如有100条数据,当前界面一次性只能显示10条,那么getCount的时候一定是100,但是,onCreateViewHolder和onBindViewHolder都只会调用10次,当你拖动的时候,才会调用第11...次。假如你再往回托,显示第一个,因为第一个之前已经创建过了,那么onCreate和OnBind都不会调用,直接显示。无论如何,当界面无法全部显示数据的时候,rcyc是不可能一次性全部调用完的,这和他的设计理念相悖了

      本文标题:ScrollView中Recyclerview显示不全

      本文链接:https://www.haomeiwen.com/subject/tugplxtx.html