KotlinとAndroidでの「パラメータTを推測するのに十分な情報がない」


110

Kotlinを使用してAndroidアプリで次のリストビューを複製しようとしています:https : //github.com/bidrohi/KotlinListView

残念ながら、自分で解決できないエラーが発生しています。これが私のコードです:

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
    }
}
}

レイアウトは次のとおりです。https//github.com/bidrohi/KotlinListView/tree/master/app/src/main/res/layout

私が取得している完全なエラーメッセージは次のとおりです: エラー:(92、31)型推論が失敗しました:funでパラメーターTを推論するのに十分な情報がありませんfindViewById(p0:Int):T!明示的に指定してください。

私が得ることができるどんな助けにも感謝します。


2
と同じように変更this.label = ... as TextViewしてみてください。これが機能するかどうか私に知らせてください。そうすれば、私はこれが適切な答えになるようにします。this.label = row?.findViewById<TextView>val listView = ...
クリスチャンブリューゲマン2017

1
どの行がエラーの原因ですか?
voddan 2017

小さな例で問題を実証できますか?
voddan 2017

@ChristianBrüggemannこのように:i.imgur.com/ZeWKjt5.pngそしてこれ:i.imgur.com/Can7w0p.png?編集すると、次のエラーが発生します:i.imgur.com/qqPAjrL.png
TimoGüntner'23

1
this.label = row?.findViewById <TextView>(R.id.label)as TextView
Alf Moh

回答:


224

APIレベル26(またはそれ以上)を使用している必要があります。このバージョンは署名を変更しましたView.findViewById()-こちらをご覧くださいhttps://developer.android.com/about/versions/oreo/android-8.0-changes#fvbi-signature

したがって、あなたの場合、結果findViewByIdが曖昧な場合は、タイプを指定する必要があります。

1 /変更

val listView = findViewById(R.id.list) as ListView

val listView = findViewById<ListView>(R.id.list)

2 /変更

this.label = row?.findViewById(R.id.label) as TextView

this.label = row?.findViewById<TextView>(R.id.label) as TextView

2 /では、rownull可能であるため、キャストが必要なだけであることに注意してください。label nullも可能だった場合、またはnullを無効にした場合は、row必要ありません。


10
これをバッチ解決する方法は次のとおりです。[ パス内で置換 ]ダイアログを開き(Ctrl + Shift + R)、正規表現ボックスをチェックします。に置き換えfindViewById\((.+?)\)\s+as\s+(.+)findViewById<$2>\($1\)、すべてのファイルに対して置換を実行します。それは私のエラーのほとんどすべてを解決しました。
グスタフカールソン2017年

1
Javaではデフォルトでビュータイプを推測するだけで十分であり、Kotlinでは不十分なのはなぜですか?findViewById(R.id.tabLayout).setOnClickListener(v-> Log.d(TAG, "login: "));これはJavaでは問題ありません。
セルジュ、

findViewById\((.+?)\)\s+as\s+([A-Za-z0-9?]+)私にとってはうまくいきます。これは、@ GustavKarlssonを超えない1行のコードを防ぎます
user3680200

1
リンクはそれを言っていません。
Alston、

7

Andoid Oは、findViewById APIを

public View findViewById(int id);

public final T findViewById(int id)

したがって、API 26をターゲットにしている場合は、

val listView = findViewById(R.id.list)as ListView

val listView = findViewById(R.id.list)

または

val listView:ListView = findViewById(R.id.list)


4

その働き

APIレベル25以下はこれを使用します

    var et_user_name = findViewById(R.id.et_user_name) as EditText

APIレベル26以上はこれを使用します

    val et_user_name: EditText = findViewById(R.id.et_user_name)

ハッピーコーディング!


2

コードをこれに変更します。主な変更が発生した場所にはアスタリスクが付いています。

package com.phenakit.tg.phenakit

import android.content.Context
import android.os.Bundle
import android.support.design.widget.BottomNavigationView
import android.support.v7.app.AppCompatActivity
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.BaseAdapter
import android.widget.ListView
import android.widget.TextView

public class MainActivity : AppCompatActivity() {

    private var mTextMessage: TextView? = null

    private val mOnNavigationItemSelectedListener = BottomNavigationView.OnNavigationItemSelectedListener { item ->
        when (item.itemId) {
            R.id.navigation_home -> {
                mTextMessage!!.setText(R.string.title_home)
                return@OnNavigationItemSelectedListener true
            }
            R.id.navigation_dashboard -> {
                mTextMessage!!.setText(R.string.title_dashboard)
                return@OnNavigationItemSelectedListener true
            }
            R.id.navigation_notifications -> {
                setContentView(R.layout.activity_list_view)
                return@OnNavigationItemSelectedListener true
            }
        }
        false
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        mTextMessage = findViewById(R.id.message) as TextView?
        val navigation = findViewById(R.id.navigation) as BottomNavigationView
        navigation.setOnNavigationItemSelectedListener(mOnNavigationItemSelectedListener)


        **val listView = findViewById<ListView>(R.id.list)**



        **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 var label: TextView

        **init { this.label = row?.findViewById<TextView?>(R.id.label) as TextView }**
    }
}

アプリでAPI 26をターゲットにしている場合、以下の答えが正解です。val listView = findViewById<ListView>(R.id.list)
EricWasTaken 2017

@ericWasTakenええ、あなたは正しいです。コンパイラは、括弧内の内容から型を推測することになっています。時々それは失敗します。あなたのコードも正しいです。変更を反映するために私の答えを更新します。
Alf Moh 2017

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.