android - RecyclerView onMeasure 性能问题
问题描述
我在 recyclerView 中使用了相当复杂的 recyclerView 项目布局。很长一段时间以来,我已经开始遇到性能问题。每次我运行应用程序时,我都会在 recyclerView 中为视图膨胀布局时收到警告。例如:
D/View: [ANR Warning]onMeasure time too long, this =android.support.constraint.ConstraintLayout{91a57db V.E...... ......I. 0,0-0,0 #7f0801d4 app:id/parentLayout}time =807 ms
D/View: [ANR Warning]onMeasure time too long, this =android.support.constraint.ConstraintLayout{20858c2 V.E...... ......I. 0,0-0,0 #7f0801d4 app:id/parentLayout}time =745 ms
D/View: [ANR Warning]onMeasure time too long, this =android.support.constraint.ConstraintLayout{880b0bc V.E...... ......I. 0,0-0,0 #7f0801d4 app:id/parentLayout}time =705 ms
每个项目平均 800 毫秒的膨胀时间真的很长。我测量onBindViewHolder()
了绑定时间,每个项目最多 10 毫秒,这很好。
有什么办法可以减少这个测量时间?我只使用ConstraintLayouts
过,因为我发现由于其快速的性能,建议对 recyclerView 项目使用约束。
此项目包含可扩展布局,该布局将在项目单击时展开。
物品布局:
<android.support.constraint.ConstraintLayout
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/parentLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<android.support.constraint.ConstraintLayout
android:id="@+id/itemInfo"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:id="@+id/imageView"
android:layout_width="80dp"
android:layout_height="80dp"
android:layout_alignParentStart="true"
android:layout_marginBottom="8dp"
android:layout_marginStart="16dp"
android:layout_marginTop="8dp"
android:visibility="visible"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_goneMarginBottom="16dp"
app:srcCompat="@mipmap/ic_launcher" />
<android.support.constraint.ConstraintLayout
android:id="@+id/itemTextLayout"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:layout_marginEnd="16dp"
android:layout_marginStart="16dp"
android:layout_marginTop="16dp"
android:visibility="visible"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toEndOf="@+id/imageView"
app:layout_constraintTop_toTopOf="parent">
<TextView
android:id="@+id/itemName"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:singleLine="false"
android:text="itemName"
android:textColor="@color/colorItemMajor"
android:textSize="16sp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<android.support.constraint.ConstraintLayout
android:id="@+id/BottomLineLayout"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginEnd="8dp"
android:layout_marginTop="4dp"
android:visibility="visible"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/itemRate_icon"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/itemName">
<ImageView
android:id="@+id/itemStatus"
android:layout_width="wrap_content"
android:layout_height="12dp"
android:layout_marginEnd="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/itemSupplier"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/itemSupplier"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:layout_marginEnd="4dp"
android:text="suppName"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/itemDel_icon"
app:layout_constraintStart_toEndOf="@id/itemStatus"
app:layout_constraintTop_toTopOf="parent" />
<ImageView
android:id="@+id/itemDel_icon"
android:layout_width="12dp"
android:layout_height="12dp"
android:layout_marginEnd="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/itemDel"
app:layout_constraintStart_toEndOf="@id/itemSupplier"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@mipmap/ic_launcher" />
<TextView
android:id="@+id/itemDel"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:layout_marginEnd="4dp"
android:text="itemDel"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/itemWT_icon"
app:layout_constraintStart_toEndOf="@+id/itemDel_icon"
app:layout_constraintTop_toTopOf="parent" />
<ImageView
android:id="@+id/itemWT_icon"
android:layout_width="12dp"
android:layout_height="12dp"
android:layout_marginStart="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toEndOf="@id/itemDel"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@mipmap/ic_launcher" />
<TextView
android:id="@+id/itemWT"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/itemWT_icon"
android:layout_alignBottom="@+id/itemWT_icon"
android:layout_gravity="fill"
android:layout_marginStart="4dp"
android:layout_toEndOf="@+id/itemWT_icon"
android:text="itemWT"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toEndOf="@+id/itemWT_icon"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
<ImageView
android:id="@+id/itemRate_icon"
android:layout_width="12dp"
android:layout_height="12dp"
android:layout_marginEnd="4dp"
android:layout_marginTop="4dp"
app:layout_constraintEnd_toStartOf="@id/itemRate"
app:layout_constraintTop_toBottomOf="@+id/itemName"
app:srcCompat="@mipmap/ic_launcher" />
<TextView
android:id="@+id/itemRate"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/itemRate_icon"
android:layout_alignBottom="@+id/itemRate_icon"
android:layout_gravity="fill"
android:layout_marginTop="4dp"
android:text="itemRate"
android:textColor="@color/colorRating"
android:textSize="10sp"
android:textStyle="bold"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toBottomOf="@+id/itemName" />
</android.support.constraint.ConstraintLayout>
</android.support.constraint.ConstraintLayout>
<android.support.constraint.ConstraintLayout
android:id="@+id/expandable_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:animateLayoutChanges="true"
android:background="@color/colorBackground"
android:orientation="horizontal"
android:visibility="visible"
app:layout_constraintTop_toBottomOf="@+id/itemInfo">
<ImageView
android:id="@+id/arrowPointer"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:srcCompat="@mipmap/ic_launcher"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/itemDesc"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:layout_marginEnd="32dp"
android:layout_marginStart="16dp"
android:layout_marginTop="24dp"
android:singleLine="false"
android:text="This is some item desc."
android:textColor="@color/colorItemMinor"
android:textSize="12sp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<ImageView
android:id="@+id/itemWeight_icon"
android:layout_width="12sp"
android:layout_height="12sp"
android:layout_marginStart="16dp"
android:layout_marginTop="20dp"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/itemDesc"
app:srcCompat="@mipmap/ic_launcher" />
<TextView
android:id="@+id/itemWeight"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/itemWeight_icon"
android:layout_alignBottom="@+id/itemWeight_icon"
android:layout_gravity="fill"
android:layout_marginStart="4dp"
android:layout_marginTop="20dp"
android:layout_toEndOf="@+id/itemWeight_icon"
android:text="itemWeight"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintStart_toEndOf="@+id/itemWeight_icon"
app:layout_constraintTop_toBottomOf="@+id/itemDesc" />
<ImageView
android:id="@+id/itemType_icon"
android:layout_width="12sp"
android:layout_height="12sp"
android:layout_marginStart="8dp"
android:layout_marginTop="20dp"
app:layout_constraintStart_toEndOf="@id/itemWeight"
app:layout_constraintTop_toBottomOf="@id/itemDesc"
app:srcCompat="@mipmap/ic_launcher" />
<TextView
android:id="@+id/itemType"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/itemType_icon"
android:layout_alignBottom="@+id/itemType_icon"
android:layout_gravity="fill"
android:layout_marginStart="4dp"
android:layout_marginTop="20dp"
android:layout_toEndOf="@+id/itemType_icon"
android:text="itemType"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintStart_toEndOf="@+id/itemType_icon"
app:layout_constraintTop_toBottomOf="@+id/itemDesc" />
<android.support.constraint.ConstraintLayout
android:id="@+id/propertyRow"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="4dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/itemType_icon">
<TextView
android:id="@+id/propertyChangeClickableText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:layout_marginStart="16dp"
android:clickable="true"
android:focusable="true"
android:text="Change Properties"
android:textColor="@color/colorPrice"
android:textSize="16sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<ImageButton
android:id="@+id/confirmItemButton"
android:layout_width="56dp"
android:layout_height="56dp"
android:layout_marginEnd="16dp"
android:background="@drawable/image_button_border_drawable"
android:scaleType="fitCenter"
app:srcCompat="@mipmap/ic_launcher"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
<View
android:id="@+id/separator"
android:layout_width="match_parent"
android:layout_height="1dp"
android:layout_marginTop="16dp"
android:background="@color/colorDivider"
app:layout_constraintTop_toBottomOf="@id/propertyRow" />
<android.support.constraint.ConstraintLayout
android:id="@+id/topRowLayoutPopup"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="8dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/separator">
<TextView
android:id="@+id/itemSupplierName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="fill"
android:layout_marginStart="16dp"
android:text="supName"
android:textColor="@color/colorItemMajor"
android:textSize="16sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<ImageView
android:id="@+id/property1_ico"
android:layout_width="16dp"
android:layout_height="16dp"
android:layout_alignBaseline="@+id/itemSupplierName"
android:layout_marginEnd="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@id/property2_ico"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@mipmap/ic_launcher" />
<ImageView
android:id="@+id/property2_ico"
android:layout_width="16dp"
android:layout_height="16dp"
android:layout_alignBaseline="@+id/itemSupplierName"
android:layout_marginEnd="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@id/property3_ico"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@mipmap/ic_launcher" />
<ImageView
android:id="@+id/property3_ico"
android:layout_width="16dp"
android:layout_height="16dp"
android:layout_alignBaseline="@+id/itemSupplierName"
android:layout_marginEnd="16dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@mipmap/ic_launcher" />
</android.support.constraint.ConstraintLayout>
<android.support.constraint.ConstraintLayout
android:id="@+id/bottomRowLayoutPopup"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="24dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/topRowLayoutPopup">
<TextView
android:id="@+id/editSupplierButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/supplierRatingBar"
android:layout_marginStart="16dp"
android:clickable="true"
android:focusable="true"
android:text="Edit Supplier"
android:textColor="@color/colorPrice"
android:textSize="16sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/supplierRating"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/supplierRatingBar"
android:layout_gravity="fill"
android:layout_marginEnd="8dp"
android:text="supplierRating"
android:textColor="@color/colorRating"
android:textSize="@dimen/food_list_bottom_row_font"
android:textStyle="bold"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@id/supplierRatingBar"
app:layout_constraintTop_toTopOf="parent" />
<RatingBar
android:id="@+id/supplierRatingBar"
style="@style/RatingBarStyle"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginEnd="8dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/supplierRatingRateNumber"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/supplierRatingRateNumber"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBaseline="@+id/supplierRatingBar"
android:layout_gravity="fill"
android:layout_marginEnd="16dp"
android:text="RateNum"
android:textColor="@color/colorItemMinor"
android:textSize="10sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
</android.support.constraint.ConstraintLayout>
解决方案
所以正如@RamiJemli 建议的那样,我用RelativeLayouts 替换了我所有的ConstraintLayouts,性能非常棒。零跳帧 - 清除 60fps 应用程序运行。显然,那些关于 RecyclerViews 中的 ConstraintLayout > RelativeLayout 的文章并不是那么正确(至少在我的应用程序中)。
推荐阅读
- excel - 在 Excel 中提取两个字符之间的特定文本字符串部分
- git - 如何诊断 git cat-file 声称缺少对象?
- c++ - 带有 std::thread 类的 C++ 堆栈泄漏
- java - 使用 @EmbeddedId 生成无效 SQL 的 CriteriaQuery selectDistinct
- compiler-construction - Given a recursive descent parser, how should I modify it for Syntax Analysis?
- reactjs - 如何将焦点移到reactjs中的错误字段
- c - 使用 memcpy 连接两个 int 数组
- youtube-iframe-api - 在 Iframe 内,Youtube 播放器取消静音功能停止工作
- javascript - 不要隐藏切换的 div
- composer-php - 有没有办法在 Windows 中设置作曲家缓存目录?