"ข้อมูลไม่เพียงพอที่จะสรุปพารามิเตอร์ T" กับ Kotlin และ Android


111

ฉันพยายามที่จะทำซ้ำ ListView ต่อไปนี้ในแอป Android ของฉันโดยใช้ Kotlin: 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) การอนุมานประเภทล้มเหลว: ข้อมูลไม่เพียงพอที่จะสรุปพารามิเตอร์ T ได้อย่างสนุกสนาน findViewById (p0: Int): T! โปรดระบุอย่างชัดเจน

ฉันขอขอบคุณสำหรับความช่วยเหลือใด ๆ ที่ฉันได้รับ


2
ลองเปลี่ยนthis.label = ... as TextViewเป็นthis.label = row?.findViewById<TextView>และทำแบบอะนาล็อกเพื่อval listView = ...? โปรดแจ้งให้เราทราบหากวิธีนี้ได้ผลเพื่อที่ฉันจะได้เป็นคำตอบที่เหมาะสมในกรณีนั้น
Christian Brüggemann

1
บรรทัดใดทำให้เกิดข้อผิดพลาด
voddan

คุณสามารถอธิบายปัญหาด้วยตัวอย่างที่เล็กกว่าได้หรือไม่?
voddan

@ ChristianBrüggemannเช่นนี้: i.imgur.com/ZeWKjt5.pngและสิ่งนี้: i.imgur.com/Can7w0p.png ? จากการแก้ไขของคุณขณะนี้มีข้อผิดพลาดเหล่านี้: i.imgur.com/qqPAjrL.png
Timo Güntner

1
ลองใช้ this.label = row? .findViewById <TextView> (R.id.label) เป็น TextView
Alf Moh

คำตอบ:


225

คุณต้องใช้ 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ไม่ได้ก็ไม่จำเป็นต้องใช้


10
วิธีแก้ปัญหานี้เป็นชุด: เปิดกล่องโต้ตอบแทนที่ในเส้นทาง (Ctrl + Shift + R) และทำเครื่องหมายที่ช่อง regex แทนที่findViewById\((.+?)\)\s+as\s+(.+)ด้วยfindViewById<$2>\($1\)และเรียกใช้การแทนที่ในไฟล์ทั้งหมด มันแก้ไขข้อผิดพลาดเกือบทั้งหมดของฉัน
Gustav Karlsson

1
เหตุใดจึงเพียงพอใน Java ที่จะอนุมานประเภท View ตามค่าเริ่มต้นและไม่เพียงพอใน Kotlin findViewById(R.id.tabLayout).setOnClickListener(v-> Log.d(TAG, "login: "));นี่ใช้ได้สำหรับ Java
uTha9

findViewById\((.+?)\)\s+as\s+([A-Za-z0-9?]+)ทำงานได้ดีขึ้นสำหรับฉัน ป้องกันไม่ให้รหัสหนึ่งบรรทัดไม่เกิน @GustavKarlsson
user3680200

1
ลิงค์ไม่ได้บอกว่า
Alston

7

Andoid O เปลี่ยน findViewById api จาก

มุมมองสาธารณะ findViewById (int id);

ถึง

สาธารณะสุดท้าย T findViewById (int id)

ดังนั้นหากคุณกำหนดเป้าหมายเป็น API 26 คุณสามารถเปลี่ยนได้

val listView = findViewById (R.id.list) เป็น 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)

Happy Coding!


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

@ericWasTaken ใช่คุณพูดถูก คอมไพเลอร์ควรจะอนุมานประเภทจากเนื้อหาในวงเล็บ บางครั้งก็ล้มเหลวที่ รหัสของคุณยังถูกต้อง จะอัปเดตคำตอบของฉันเพื่อแสดงการเปลี่ยนแปลง
Alf Moh

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.