我似乎无法理解 Kotlin 泛型,请帮忙
我读了here和 here而且我仍然无法确定如何进行这项工作。我有一个 RecyclerView 适配器,它使用抽象的父 BaseFieldVH 作为 ViewHolder 类。
所以 Adapter 类看起来像这样:
class MyAdapter() : RecyclerView.Adapter<BaseFieldVH<*>>() {
...
override fun onBindViewHolder(holder: BaseFieldVH<*>, position: Int) {
holder.bind(data[position])
}
}
ViewHolder 的实现如下所示:
abstract class BaseFieldVH<in FF : BaseFormField>
(itemView: View) : RecyclerView.ViewHolder(itemView) {
...
abstract fun bind(formField: FF)
}
但实际调用holder.bind(data[position])
正在显示错误:
Out-projected type 'BaseFieldVH<*>' prohibits the use of 'public
abstract fun bind(formField: FF): Unit defined in ...
什么都不用?
我还尝试用 BaseFieldVH<in Nothing>
定义适配器但随后我收到类型不匹配错误,因为我试图将 Nothing 放入需要 BaseFormField 的函数中。
使用 BaseFormField
使用 BaseFieldVH<iBaseFormField>
定义适配器实际上解决了绑定(bind)问题,但是在 onCreateViewHolder 中, View 持有者的类型不匹配:
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): BaseFieldVH<BaseFormField> {
val itemView: View
when (viewType) {
HEADER_TYPE -> {
itemView = LayoutInflater.from(parent.context)
.inflate(R.layout.item_header, parent, false)
return HeaderVH(itemView)
}
HeaderVH 是 BaseFieldVH 的扩展,因此存在类型不匹配
最佳答案
看起来这个问题的答案是您不能实现指定绑定(bind)泛型的 ViewHolder 类。我猜这是因为 RecyclerView.Adapter 是这样写的在 Java 中,任何 Kotlin 扩展都过于严格,无法正确实现。
如果你能反驳这个结论,请告诉我!
我找到了 this帖子以我能理解的方式解释了整个投影概念。
最后我放弃了控制 ViewHolder 类型的尝试:
abstract class BaseFieldVH
(itemView: View) : RecyclerView.ViewHolder(itemView) {
...
abstract fun bind(formField: Any)
}
这意味着 BaseFieldVH 的每个子类都必须在其绑定(bind)函数中进行一些类型检查。例如:
class HeaderVH
(itemView: View) : BaseFieldVH(itemView) {
...
override fun bind(headerField: Any) {
if (headerField is HeaderField) {
// do something
}
}
但现在至少我没有收到任何错误。
关于android - 外投影类型禁止使用方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51049049/