ViewBinding 不适用于 RecyclerView 项目中包含的布局
ViewBinding not working for layout included inside RecyclerView item
这是我的 RecyclerView 项目布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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="wrap_content"
android:orientation="vertical"><!--
android:background="@color/home_item_bg"-->
<include
android:id="@+id/layoutSectionHeader"
layout="@layout/section_header_rv_item_home"/>
<androidx.cardview.widget.CardView
style="@style/Widget.MaterialComponents.CardView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardElevation="2dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_horizontal"
android:orientation="vertical"
android:paddingStart="16dp"
android:paddingTop="8dp"
android:paddingEnd="16dp"
android:paddingBottom="8dp">
<TextView
android:id="@+id/tvMarketAndTerm"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textStyle="bold"
tools:text="NASDAQ - Medium Term" />
<com.github.mikephil.charting.charts.PieChart
android:id="@+id/pieChart"
android:layout_width="match_parent"
android:layout_height="@dimen/pie_chart_height" />
</LinearLayout>
</androidx.cardview.widget.CardView>
</LinearLayout>
这是section_header_rv_item_home.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/llSectionHeader"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_vertical"
android:orientation="horizontal"
android:paddingStart="16dp"
android:paddingTop="12dp"
android:background="?selectableItemBackground"
android:paddingEnd="16dp"
android:paddingBottom="12dp"><!--
android:background="@drawable/ripple_white_light_grey"-->
<TextView
android:id="@+id/tvProductName"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ellipsize="end"
android:maxLines="1"
android:textAllCaps="false"
android:textColor="@color/gray"
android:textSize="20sp"
android:textStyle="bold"
tools:text="@string/stock_exchange_barometer" />
<TextView
android:id="@+id/tvSeeAll"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/see_all"
android:textColor="@color/colorAccent"
android:textSize="13sp"
android:textStyle="bold"
android:visibility="gone"
tools:visibility="visible" />
</LinearLayout>
这是访问包含的布局中的视图的方式
inner class BarometerVH(private val rvBinding: RvItemHomeSeBarometerBinding) :
RecyclerView.ViewHolder(rvBinding.root) {
fun bind(homeItem: HomeItem) {
rvBinding.layoutSectionHeader.tvProductName.text = homeItem.title
}
}
我没有编译时错误,但是当我 运行 应用
时出现这个异常
java.lang.NullPointerException: Missing required view with ID: com.example.appname:id/layoutSectionHeader
在 Activity 或 Fragment 中包含布局对我有用,但在 Recycler View 中使用时不起作用。
我尝试了此论坛中针对类似异常提供的多种解决方案,但 none 适用于回收站视图,而不是关于使用
编辑 1:我添加了来自 RecyclerView Adapter 的覆盖方法
override fun onCreateViewHolder(
parent: ViewGroup,
viewType: Int
) = MarketCommentaryVH(RvItemHomeMarketCommentaryBinding.bind(parent))
override fun onBindViewHolder(
holder: RecyclerView.ViewHolder,
position: Int
) {
holder.bind(homeItemList[position])
}
A RecyclerView
通过创建和重用一堆 ViewHolder
来工作,这些 ViewHolder
包含一个视图布局以显示项目的详细信息。您在 onCreateViewHolder
方法中设置这些,您:
- 为项目扩充布局
- 将膨胀的视图传递给
ViewHolder
构造函数
你实际上并没有膨胀布局,你正在做
RvItemHomeMarketCommentaryBinding.bind(parent)
而不是
RvItemHomeMarketCommentaryBinding.inflate(LayoutInflater.from(context), parent, false)
(你需要在给 ViewHolder
的布局充气时传入 parent
,这样它才能以正确的尺寸自行布局等)
Binding
组件上的 bind
函数采用已经膨胀的视图层次结构,并尝试查找具有它知道的 ID 的所有组件。因此,例如,它会尝试查找 layoutSectionHeader
,以便可以将 binding.layoutSectionHeader
分配给该视图。
问题是,在您传入的 parent
布局中不存在,因为那只是 RecyclerView
的布局,它不包含来自 ViewHolder
布局。所以 bind
不起作用 - 视图无法绑定。您需要调用 inflate
而不是
这是我的 RecyclerView 项目布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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="wrap_content"
android:orientation="vertical"><!--
android:background="@color/home_item_bg"-->
<include
android:id="@+id/layoutSectionHeader"
layout="@layout/section_header_rv_item_home"/>
<androidx.cardview.widget.CardView
style="@style/Widget.MaterialComponents.CardView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardElevation="2dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_horizontal"
android:orientation="vertical"
android:paddingStart="16dp"
android:paddingTop="8dp"
android:paddingEnd="16dp"
android:paddingBottom="8dp">
<TextView
android:id="@+id/tvMarketAndTerm"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textStyle="bold"
tools:text="NASDAQ - Medium Term" />
<com.github.mikephil.charting.charts.PieChart
android:id="@+id/pieChart"
android:layout_width="match_parent"
android:layout_height="@dimen/pie_chart_height" />
</LinearLayout>
</androidx.cardview.widget.CardView>
</LinearLayout>
这是section_header_rv_item_home.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/llSectionHeader"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_vertical"
android:orientation="horizontal"
android:paddingStart="16dp"
android:paddingTop="12dp"
android:background="?selectableItemBackground"
android:paddingEnd="16dp"
android:paddingBottom="12dp"><!--
android:background="@drawable/ripple_white_light_grey"-->
<TextView
android:id="@+id/tvProductName"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ellipsize="end"
android:maxLines="1"
android:textAllCaps="false"
android:textColor="@color/gray"
android:textSize="20sp"
android:textStyle="bold"
tools:text="@string/stock_exchange_barometer" />
<TextView
android:id="@+id/tvSeeAll"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/see_all"
android:textColor="@color/colorAccent"
android:textSize="13sp"
android:textStyle="bold"
android:visibility="gone"
tools:visibility="visible" />
</LinearLayout>
这是访问包含的布局中的视图的方式
inner class BarometerVH(private val rvBinding: RvItemHomeSeBarometerBinding) :
RecyclerView.ViewHolder(rvBinding.root) {
fun bind(homeItem: HomeItem) {
rvBinding.layoutSectionHeader.tvProductName.text = homeItem.title
}
}
我没有编译时错误,但是当我 运行 应用
时出现这个异常java.lang.NullPointerException: Missing required view with ID: com.example.appname:id/layoutSectionHeader
在 Activity 或 Fragment 中包含布局对我有用,但在 Recycler View 中使用时不起作用。
我尝试了此论坛中针对类似异常提供的多种解决方案,但 none 适用于回收站视图,而不是关于使用 编辑 1:我添加了来自 RecyclerView Adapter 的覆盖方法override fun onCreateViewHolder(
parent: ViewGroup,
viewType: Int
) = MarketCommentaryVH(RvItemHomeMarketCommentaryBinding.bind(parent))
override fun onBindViewHolder(
holder: RecyclerView.ViewHolder,
position: Int
) {
holder.bind(homeItemList[position])
}
A RecyclerView
通过创建和重用一堆 ViewHolder
来工作,这些 ViewHolder
包含一个视图布局以显示项目的详细信息。您在 onCreateViewHolder
方法中设置这些,您:
- 为项目扩充布局
- 将膨胀的视图传递给
ViewHolder
构造函数
你实际上并没有膨胀布局,你正在做
RvItemHomeMarketCommentaryBinding.bind(parent)
而不是
RvItemHomeMarketCommentaryBinding.inflate(LayoutInflater.from(context), parent, false)
(你需要在给 ViewHolder
的布局充气时传入 parent
,这样它才能以正确的尺寸自行布局等)
Binding
组件上的 bind
函数采用已经膨胀的视图层次结构,并尝试查找具有它知道的 ID 的所有组件。因此,例如,它会尝试查找 layoutSectionHeader
,以便可以将 binding.layoutSectionHeader
分配给该视图。
问题是,在您传入的 parent
布局中不存在,因为那只是 RecyclerView
的布局,它不包含来自 ViewHolder
布局。所以 bind
不起作用 - 视图无法绑定。您需要调用 inflate
而不是