“Pas assez d'informations pour en déduire le paramètre T” avec Kotlin et Android
Je suis en train de reproduire le suivant ListView dans mon application Android à l'aide de Kotlin: https://github.com/bidrohi/KotlinListView.
Malheureusement, je reçois une erreur, je suis incapable de résoudre moi-même.
Voici mon code:
MainActivity.kt:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
val listView = findViewById(R.id.list) as ListView
listView.adapter = ListExampleAdapter(this)
}
private class ListExampleAdapter(context: Context) : BaseAdapter() {
internal var sList = arrayOf("Eins", "Zwei", "Drei")
private val mInflator: LayoutInflater
init {
this.mInflator = LayoutInflater.from(context)
}
override fun getCount(): Int {
return sList.size
}
override fun getItem(position: Int): Any {
return sList[position]
}
override fun getItemId(position: Int): Long {
return position.toLong()
}
override fun getView(position: Int, convertView: View?, parent: ViewGroup): View? {
val view: View?
val vh: ListRowHolder
if(convertView == null) {
view = this.mInflator.inflate(R.layout.list_row, parent, false)
vh = ListRowHolder(view)
view.tag = vh
} else {
view = convertView
vh = view.tag as ListRowHolder
}
vh.label.text = sList[position]
return view
}
}
private class ListRowHolder(row: View?) {
public val label: TextView
init {
this.label = row?.findViewById(R.id.label) as TextView
}
}
}
Les mises en page sont exactement comme ici: https://github.com/bidrohi/KotlinListView/tree/master/app/src/main/res/layout
Le plein de message d'erreur que j'obtiens est ceci:
Erreur:(92, 31) l'inférence de Type a échoué: Pas assez d'information pour en déduire le paramètre T dans le plaisir findViewById(p0: Int): T!
Merci de le préciser explicitement.
J'apprécierais toute aide que je peux obtenir.
- Pouvez-vous essayer de changer de
this.label = ... as TextView
àthis.label = row?.findViewById<TextView>
et de le faire d'une façon analogue, pourval listView = ...
? S'il vous plaît laissez-moi savoir si cela fonctionne si je peux faire de cette une réponse adéquate dans ce cas. - La ligne qui provoque une erreur?
- Pouvez-vous démontrer le problème avec un petit exemple?
- Comme ceci: i.imgur.com/ZeWKjt5.png et ça: i.imgur.com/Can7w0p.png ? Avec vos modifications il y a maintenant ces erreurs: i.imgur.com/qqPAjrL.png
- Cette ligne provoque l'erreur: cette.label = ligne?.findViewById(R. id.étiquette) TextView
- Essayez cette de cette.label = ligne?.findViewById<TextView>(R. id.étiquette) TextView
- L'application démarre mais se bloque tout de suite après. L'erreur dans la console de débogage est maintenant: kotlin.TypeCastException: null ne peut pas être jeté à la non-nulle de type android.widget de.ListView
- Ajouter un point d'interrogation à la fin de TextView
- Avec un point d'interrogation à la fin (ce.label = ligne?.findViewById<TextView>(R. id.étiquette) TextView?) J'obtiens l'erreur suivante: Erreur:(94, 26) incompatibilité de Type: type inféré est TextView? mais TextView qui était attendu
- quoi à ce sujet? public val étiquette: TextView? init { ce.label = ligne?.findViewById<TextView?>(R. id.étiquette) TextView? }
- Malheureusement, même message d'erreur: Erreur:(94, 26) incompatibilité de Type: type inféré est TextView? mais TextView était attendu. Et ce dans l'Android Moniteur: java.lang.RuntimeException: Impossible de démarrer l'activité ComponentInfo{com.phenakit.tg.phenakit/com.phenakit.tg.phenakit.MainActivity}: kotlin.TypeCastException: null ne peut pas être jeté à la non-nulle de type android.widget de.ListView @AlfMoh
- Laissez-nous continuer cette discussion dans le chat.
Vous devez vous connecter pour publier un commentaire.
Vous devez être à l'aide de l'API de niveau 26 (ou ci-dessus). Cette version a changé la signature de
View.findViewById()
- voir ici https://developer.android.com/preview/api-overview.html#fvbi-signatureDonc dans votre cas, où le résultat de
findViewById
est ambigu, vous devez fournir le type:1/Changement
val listView = findViewById(R.id.list) as ListView
àval listView = findViewById<ListView>(R.id.list)
2/Changement
this.label = row?.findViewById(R.id.label) as TextView
àthis.label = row?.findViewById<TextView>(R.id.label) as TextView
Noter que dans le 2, la distribution des rôles est nécessaire uniquement parce que
row
est nullable. Silabel
a nullable trop, ou si vous avez fait
row
accepte pas la valeur null, il ne serait pas nécessaire.findViewById\((.+?)\)\s+as\s+(.+)
avecfindViewById<$2>\($1\)
et exécuter le remplacer dans tous les fichiers. Il a résolu la quasi-totalité de mes erreurs.findViewById(R.id.tabLayout).setOnClickListener(v-> Log.d(TAG, "login: "));
c'est OK pour Java.findViewById\((.+?)\)\s+as\s+([A-Za-z0-9?]+)
fonctionne le mieux pour moi. Il empêche une ligne de code n'est pas sur @GustavKarlssonAndoid O changement findViewById api de
à
donc, si vous êtes la cible d'API 26, vous pouvez modifier
à
ou
Son Travail
API de Niveau 25 ou au-dessous de l'utilisation de ce
API de Niveau 26 ou au-Dessus de l'utilisation de ce
Bon Codage !
Modifier votre code pour cela. Où les principaux changements survenus sont marquées par des astérisques.
val listView = findViewById<ListView>(R.id.list)
Je vous suggère d'utiliser
synthetics
kotlin Android extension:https://kotlinlang.org/docs/tutorials/android-plugin.html
https://antonioleiva.com/kotlin-android-extensions/
Dans votre cas, le code sera quelque chose comme ceci:
Aussi simple que ça 😉