我们有一个NestedScrollView,它包含两个不同的RecyclerView
,它们都使用垂直滚动。滚动布局位于SwipeRefreshLayout
内。
更新:我们了解getItemViewType(pos)
方法并在其他地方使用它。但是在这里我们有复杂的逻辑,将进一步划分为2个不同的屏幕,因此我们为RecyclerView中的每一个都有两个单独的演示者,并将它们合并为一个并且在一个月内再次分离不是一个选项。
从某个时刻起,我们注意到使用刷卡更新屏幕会开始冻结UI:日志中跳过的帧,无法使用UI进行任何操作,冻结进度条。它需要长达五秒钟,但我们没有得到ANR。它在开始时效果很好,最后冻结。如果我们删除适配器中的notifyDataSetChanged()
,一切看起来都不错。
我们尝试删除其中一个回收站视图,从ViewHolder.onCreate()
和onBindViewHolder()
删除所有操作,但它没有帮助。此外,我们交叉检查问题是否来自某些数据处理(我们使用RxJava 2来操纵线程),我们看到所有具有网络,数据库和数据处理的操作都是在非UI线程中完成的,并且已经完成了当滞后开始时。
这是布局:
<android.support.v4.widget.SwipeRefreshLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/swipe_refresh"
android:layout_width="match_parent"
android:layout_height=«match_parent"
android:orientation="vertical"
>
<android.support.v4.widget.NestedScrollView
android:layout_width="match_parent"
android:layout_height="wrap_content»
>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation=«vertical"
android:paddingBottom="@dimen/spacing_bigger»>
<android.support.v7.widget.RecyclerView
android:id="@+id/recycle_chats"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/white_bg"
/>
<android.support.v7.widget.RecyclerView
android:id="@+id/recycle_friends"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/white_bg"
/>
</LinearLayout>
</android.support.v4.widget.NestedScrollView>
</android.support.v4.widget.SwipeRefreshLayout>
答案 0 :(得分:1)
看起来NestedScrollView
layout_height="wrap_content"
打破了所有儿童RecyclerViews
的优化。无论具有SwipeRefreshLayout
的父视图(layout_height="match_parent"
)的约束如何,它都会为它们提供无限空间。因此,RecyclerView
会立即为适配器中的所有模型创建视图持有者。记录显示,它们每个需要20-60毫秒才能为100行带来2-5秒滞后。
我们在onCreateViewHolder()中添加了日志记录,并注意到它有几十(几乎一百)个调用,而屏幕上只有大约十个元素。将NestedScrollView
的{{1}}更改为layout_height
解决了问题。
答案 1 :(得分:1)
我也在努力解决这个问题,终于明白了我无法在nestedscrollView WITH notifyData中使用RecyclerView,所以我这样做了:
我将CollapsingToolbarLayout放在CoordinatorLayout内的AppBarLayout中。 然后我将RecyclerView标头放在CollapsingToolbarLayout中,将RecyclerView放在AppBarLayout下面的CoordinatorLayout中。
我还使用了RecyclerView的addOnScrollListener方法在分页后为RecyclerView加载新项目。
这是我的xml文件:
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/
android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#ffffff">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.design.widget.CoordinatorLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.design.widget.AppBarLayout
android:id="@+id/activity_txt_news_detail.app_bar_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#00ffffff"
app:elevation="0dp">
<android.support.design.widget.CollapsingToolbarLayout
android:id="@+id/collapsingToolbarLayout"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:expandedTitleMarginStart="64dp"
app:layout_scrollFlags="scroll|snap">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
// all other views
// as header of
// recyclerview comes here
</LinearLayout>
<!--todo tgs : here top-->
</android.support.design.widget.CollapsingToolbarLayout>
</android.support.design.widget.AppBarLayout>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
app:layout_behavior="@string/
appbar_scrolling_view_behavior">
<android.support.v7.widget.RecyclerView
android:id="@+id/home_news_list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
tools:listitem="@layout/row_news_txt_small" />
</RelativeLayout>
</android.support.design.widget.CoordinatorLayout>
</LinearLayout>
</FrameLayout>
这行代码app:layout_behavior =&#34; @ string / appbar_scrolling_view_behavior&#34; 应该是在recyclerview容器布局中非常重要
如果它对您有用,请通知我。一切顺利
答案 2 :(得分:0)
为什么要添加nestedScrollLayout和linearLayout以及2个recyclerViews。
只需使用swipeRefreshLayout和RecyclerView
即可实现所有这一切尝试查看如何根据数据的位置或类型添加不同的视图。所以技术上你只有一个recyclerView和一个适配器,但适配器将处理两个案例(朋友和聊天)并将数据附加到视图(recyclerView)。
由于RecyclerView还扩展了NestedScrollView,在NestedScrollView中保留一个RecyclerView会使你的滚动抖动,然后你将不得不添加isScrollContainer = true等标志。所以不要这样做!