Android ViewHolder不能强制转换为其他ViewHolder

Android ViewHolder不能强制转换为其他ViewHolder,android,kotlin,android-recyclerview,custom-adapter,Android,Kotlin,Android Recyclerview,Custom Adapter,我的Recyclerview的自定义适配器有问题。 我试图在一个回收器视图中显示两个不同的视图和不同的数据源。 在我做一些改变之前它工作得很好,现在它得到了一个例外 它工作正常,直到notifyDataSetChanged()函数! 当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃 我找了很多,但找不到答案。谢谢你的帮助 我的自定义适配器: import android.app.Activity import android.

我的Recyclerview的自定义适配器有问题。 我试图在一个回收器视图中显示两个不同的视图和不同的数据源。 在我做一些改变之前它工作得很好,现在它得到了一个例外

它工作正常,直到notifyDataSetChanged()函数! 当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃

我找了很多,但找不到答案。谢谢你的帮助

我的自定义适配器:

import android.app.Activity
import android.content.Context
import android.content.DialogInterface
import android.content.Intent
import android.os.Vibrator
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.*
import androidx.cardview.widget.CardView
import androidx.recyclerview.widget.RecyclerView
import artemis.team.picandtext.*
import com.squareup.picasso.Picasso
import org.jetbrains.anko.alert
import org.jetbrains.anko.sdk27.coroutines.onClick
import java.util.*

class RecAdapter(
    context: Context,
    activity: Activity,
    grplist: MutableList<GrpsValues> = mutableListOf(),
    itemslist: MutableList<ItemsValues> = mutableListOf()
) : RecyclerView.Adapter<RecyclerView.ViewHolder>() {
    private var context: Context? = null
    private var activity: Activity? = null
    private var grpList = mutableListOf<GrpsValues>()
    private var itemsList = mutableListOf<ItemsValues>()
    private lateinit var tts: TextToSpeech
    private var voiceText = "Text not found!"
    private var voiceLocale = "en-US"
    private var mode = 0
    private var itemsSize = 0
    private var groupsSize = 0
    private val ITEM = 100
    private val GROUP = 200
    private var HOLDER = GROUP

    init {
        this.activity = activity
        this.context = context
        this.grpList = grplist
        this.itemsList = itemslist
        itemsSize = itemsList.size
        groupsSize = grpList.size
    }

    override fun getItemCount(): Int {
        return (itemsList.size + grpList.size)
    } // getItemCount

    override fun onBindViewHolder(recHolder: RecyclerView.ViewHolder, pos: Int) {

        Mode()

        if (mode == ITEM && HOLDER == ITEM) {

            val holder = recHolder as ItemsViewHolder

            if (itemsList.size > 0) {
                val position = pos - grpList.size

                Picasso.get()
                    .load(itemsList[position].url)
                    .error(R.drawable.empty)
                    .into(holder.image)

                holder.title.text = itemsList[position].title
                holder.txtID.text = itemsList[position].id
                holder.grpID.text = itemsList[position].grpID

                val grpID = holder.grpID.text.toString()

                lateinit var dialog: DialogInterface

                holder.layout.onClick {
                    (context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
                    dialog = context!!.alert {
                        val view = activity!!.layoutInflater.inflate(R.layout.dialog_item_details, null)
                        val prgDatabase = PrgDatabase(context)
                        val value = prgDatabase.ItemsFind(holder.txtID.text.toString())
                        view.findViewById<TextView>(R.id.TxtText).text = value.title
                        Picasso.get()
                            .load(value.url)
                            .error(R.drawable.empty)
                            .into(view.findViewById<ImageView>(R.id.imageView))

                        customView = view
                    }.show()
                }

                holder.txtDelete.onClick {
                    (context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
                    context!!.alert {
                        title = "ItemsDelete"
                        message = "ItemsDelete item from the list?"

                        positiveButton("Yup") {
                            val prgDatabase = PrgDatabase(context)
                            prgDatabase.ItemsDelete(holder.txtID.text.toString())
                            itemsList = PrgDatabase(context).ItemsGetAll(grpID)

                            this@RecAdapter.notifyDataSetChanged()

                            it.dismiss()
                        }

                        negativeButton("NO") {
                            it.dismiss()
                        }

                    }.show()
                }

                holder.txtEdit.onClick {
                    (context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)

                    dialog = context!!.alert {
                        val view = activity!!.layoutInflater.inflate(R.layout.dialog_items, null)
                        view.findViewById<Button>(R.id.BTNsaveID).text = "Save changes"
                        val prgDatabase = PrgDatabase(context)

                        val adapter = ArrayAdapter<String>(
                            context!!,
                            android.R.layout.simple_spinner_item,
                            context!!.resources.getStringArray(
                                R.array.langueges
                            )
                        )
                        adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item)
                        view.findViewById<Spinner>(R.id.SpLangsID).adapter = adapter

                        val value = prgDatabase.ItemsFind(holder.txtID.text.toString())
                        view.findViewById<EditText>(R.id.ETtitleID).setText(value.title)
                        view.findViewById<EditText>(R.id.ETurlID).setText(value.url)
                        view.findViewById<Spinner>(R.id.SpLangsID).setSelection(
                            when (value.local) {
                                "English" -> 0
                                "Russian" -> 1
                                "Turkish" -> 2
                                "French" -> 3
                                "German" -> 4
                                "Italian" -> 5
                                "Korean" -> 6
                                else -> 0
                            }
                        )
                        view.findViewById<Button>(R.id.BTNsaveID).onClick {
                            if (view.findViewById<EditText>(R.id.ETtitleID).text.isNotEmpty()
                                && view.findViewById<EditText>(R.id.ETurlID).text.isNotEmpty()
                            ) {
                                val value = ItemsValues(
                                    title = view.findViewById<EditText>(R.id.ETtitleID).text.toString(),
                                    url = view.findViewById<EditText>(R.id.ETurlID).text.toString(),
                                    id = holder.txtID.text.toString(),
                                    grpID = grpID,
                                    local = view.findViewById<Spinner>(R.id.SpLangsID).selectedItem.toString()
                                )
                                prgDatabase.ItemsUpdate(
                                    value
                                )
                                itemsList[position] = value
                                dialog.dismiss()


                                this@RecAdapter.notifyDataSetChanged()
                                this@RecAdapter.notifyItemChanged(position)


                            }// if
                        } // btn on click
                        customView = view

                    }.show()
                } // on click
            }

        } // if Items
        else if (mode == GROUP && HOLDER == GROUP) {

            val holder = recHolder as GrpsViewHolder
            if (grpList.size > 0) {

                val position = pos

                holder.titleGrp.text = grpList[position].title

                holder.txtItemIDGrp.text = grpList[position].id
                holder.txtParentGrpIDGrp.text = grpList[position].parent

                holder.layoutGrp.onClick {
                    val intent = Intent(context, MainActivity::class.java)
                    intent.putExtra("GrpID", holder.txtItemIDGrp.text.toString())
                    intent.putExtra("GrpTitle", grpList[position].title)
                    activity!!.startActivity(intent)
                }

                holder.txtDeleteGrp.onClick {
                    (context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
                    context!!.alert {
                        title = "Delete"
                        message = """
                        Delete Group from the list?

                        * With delete this group, all it's words and pictures will be removed!
                        """.trimIndent()

                        positiveButton("Yup") {
                            val prgDatabase = PrgDatabase(context)
                            prgDatabase.GrpsDelete(holder.txtItemIDGrp.text.toString())

                            grpList = PrgDatabase(context).GrpsGetAll(grpList[position].parent)

                            this@RecAdapter.notifyItemRemoved(position)
                            this@RecAdapter.notifyDataSetChanged()

                            it.dismiss()
                        }

                        negativeButton("NO") {
                            it.dismiss()
                        }

                    }.show()
                }

                lateinit var dialog: DialogInterface
                holder.txtEditGrp.onClick {
                    (context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)

                    dialog = context!!.alert {
                        val view = activity!!.layoutInflater.inflate(R.layout.dialog_grps, null)

                        view.findViewById<Button>(R.id.BTNsaveID).text = "Save changes"
                        val prgDatabase = PrgDatabase(context)
                        val value = prgDatabase.GrpsFind(holder.txtItemIDGrp.text.toString())
                        view.findViewById<EditText>(R.id.ETtitleID).setText(value.title)
                        view.findViewById<Button>(R.id.BTNsaveID).onClick {
                            if (view.findViewById<EditText>(R.id.ETtitleID).text.isNotEmpty()) {
                                val value = GrpsValues(
                                    title = view.findViewById<EditText>(R.id.ETtitleID).text.toString(),
                                    id = holder.txtItemIDGrp.text.toString(),
                                    parent = holder.txtParentGrpIDGrp.text.toString()
                                )
                                prgDatabase.GrpsUpdate(
                                    value
                                )
                                grpList[position] = value
                                dialog.dismiss()

                                this@RecAdapter.notifyDataSetChanged()
                                this@RecAdapter.notifyItemChanged(position)

                            }// if
                        } // btn on click
                        customView = view
                    }.show()
                } // on click
            } // if
        } // GROUP

    } // onBindViewHolder

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
        if (groupsSize > 0) {
            HOLDER = GROUP
            return GrpsViewHolder(
                LayoutInflater.from(parent.context).inflate(
                    R.layout.custom_rec_grps_layout,
                    parent,
                    false
                )
            )

        } else {
            HOLDER = ITEM
            return ItemsViewHolder(
                LayoutInflater.from(parent.context).inflate(
                    R.layout.custom_rec_items_layout,
                    parent,
                    false
                )
            )
        }
    }

    inner class ItemsViewHolder(view: View) : RecyclerView.ViewHolder(view) {
        val image: ImageView = view.findViewById(R.id.imageView)
        val title: TextView = view.findViewById(R.id.TxtText)
        val txtID: TextView = view.findViewById(R.id.TxtItemID)
        val grpID: TextView = view.findViewById(R.id.TxtGrpID)
        val txtDelete: TextView = view.findViewById(R.id.TxtDeleteID)
        val txtEdit: TextView = view.findViewById(R.id.TxtEditID)
        val layout: CardView = view.findViewById(R.id.ConstItemsID)
    }

    inner class GrpsViewHolder(view: View) : RecyclerView.ViewHolder(view) {
        val titleGrp: TextView = view.findViewById(R.id.GrpTxtText)
        val txtItemIDGrp: TextView = view.findViewById(R.id.GrpTxtItemID)
        val txtParentGrpIDGrp: TextView = view.findViewById(R.id.GrpTxtGrpID)
        val txtDeleteGrp: TextView = view.findViewById(R.id.GrpTxtDeleteID)
        val txtEditGrp: TextView = view.findViewById(R.id.GrpTxtEditID)
        val layoutGrp: CardView = view.findViewById(R.id.CnstGrpsID)

    }

    override fun getItemViewType(position: Int): Int {
        return position
    }


    fun Mode() {
        if (groupsSize > 0) {
            groupsSize--
            mode = GROUP
        } else if (itemsSize > 0) {
            itemsSize--
            mode = ITEM
        }
    }
}

不能将ViewHolder类型的一个子类强制转换为ViewHolder类的另一个类型,因为它们不是同一类型

    ViewHolder
       /\
RecHolder ItemsViewHolder
RecHolder
是一个视图持有者,
ItemsViewHolder
是一个视图持有者,但仅仅因为它们共享同一个父对象,并不能使
RecHolder
成为
ItemsViewHolder
,所以演员阵容永远不会成功

检查您的逻辑以确定您的
onCreateViewHolder
返回的ViewHolder,因为这是进入
onBindViewHolder的
ViewHolder
类型


如果你想要一种方法,在你投下每一个之前,在你的逻辑内部进行双重检查,那么做
If(viewHolder是RecHolder)
If(viewHolder是ItemViewHolder
)来验证你得到的是你认为得到的那一个。

你能为
GrpsViewHolder
@BlackHatSamurai发布你的代码吗。在getItemViewType函数之前。在notifyDataSetChanged()函数之前,它工作正常!当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃!同一类演员例外?是的!现在,我改变了逻辑,正如你所说,现在它的工作没有崩溃!但是当删除一个组时,它会完全从db中删除,但在recyclerview中,它会删除最后一项。关闭并重新打开应用程序后显示正常。这是因为在更新数据库后,您可能没有填充recyclerview的数据。您正在将itemsList声明为private`private var itemsList=mutableListOf()`并且您没有在任何地方使用该列表中的项的方法。这意味着您只在每次重新创建适配器时获得新数据。`init{this.activity=activity this.context=context this.grpList=grpList this.itemsList=itemsList}`您还可以不必分配这些作为参数传入的值,通过将
var
关键字添加到构造函数类RecAdapter(var-context:context,var-activity:activity
    ViewHolder
       /\
RecHolder ItemsViewHolder