130

我正在尝试使用 Kotlin 在我的 Android 应用程序中复制以下 ListView: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

我收到的完整错误消息是: Error:(92, 31) Type inference failed: Not enough information to infer parameter T in fun findViewById(p0: Int): T! 请明确指定。

我会很感激我能得到的任何帮助。

4

6 回答 6

249

您必须使用 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/ 中,仅需要强制转换,因为row它可以为空。如果它也可以label 为空,或者如果您设置为row不可为空,则不需要它。

于 2017-07-25T08:16:13.750 回答
8

Andoid O 更改 findViewById api 从

公共视图 findViewById(int id);

公共最终 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)

于 2017-08-03T15:34:36.297 回答
6

它的工作

API Level 25 或以下使用此

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

API Level 26 或以上使用这个

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

快乐编码!

于 2017-11-13T11:57:21.037 回答
4

将您的代码更改为此。发生主要变化的地方用星号标记。

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 }**
    }
}
于 2017-07-23T20:26:16.680 回答
1

我建议你使用syntheticskotlin Android 扩展:

https://kotlinlang.org/docs/tutorials/android-plugin.html

https://antonioleiva.com/kotlin-android-extensions/

在您的情况下,代码将是这样的:

init {
   this.label = row.label
}

就如此容易 ;)

于 2017-12-12T13:35:55.607 回答
1

在 android 12 中,从您的代码中删除as并将您的modelin <> 放在getParcelableExtra.

改变

intent.getParcelableExtra("MyModel") as MyModel

intent.getParcelableExtra `<MyModel>` ("MyModel")!!
于 2022-01-26T10:50:23.647 回答