3

我已经开始学习在android中使用kotlin语言并在初始化我的按钮变量时遇到问题,因为在定义我的变量时,当我用空值初始化并在oncreate函数中绑定变量时它要求给出一些初始值它给出

kotlin.KotlinNullPointerException

这是我的代码

class AddsFragment : Fragment() {

    var Add: Button = null!!

    override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        val Rootview = inflater!!.inflate(R.layout.clubsfragment, null, false)
        Add = Rootview.findViewById(R.id.add) as Button
        return Rootview
    }
}
4

3 回答 3

16

!!操作员检查接收者是否是以及null是否是 throws KotlinNullPointerException。所以null!!总会抛出异常。

您可以通过以下方式实现您想要的:

  1. 将属性的类型设置为Button?。在这种情况下,当访问按钮的方法时,您将不得不使用?!!

    var add: Button? = null
    // Initialize it somewhere.
    
    add?.setText("Text") // Calls setText if Add != null
    add!!.setText("Text") // Throws an exception if Add == null
    
  2. 使按钮成为lateinit属性。

    lateinit var add: Button
    
  3. 使按钮成为notNull委托属性。

    var add: Button by Delegates.notNull()
    

在最后两种情况下,您无法检查按钮是否为null。如果您需要null比较来处理变量,请使用第一种方法。


还有另一种方法,我不会在这个答案中详细描述。首先是使用Kotlin Android Extensions。这是一个编译器插件,可为您的视图生成合成属性,因此您无需调用findViewById()并可以使用生成的属性访问视图。

第二种方法是创建您自己的代表来调用findViewById()您。它可能看起来像这样:

val add: Button by bindView(R.id.add)

您可以在KotterKnife项目中找到此类委托的示例。

于 2016-05-29T19:08:01.427 回答
4

您可以使用以下任何一种方法在 kotlin 中初始化视图:

1. 可空变量

private var textView: TextView? = null
…
textView = findViewById(R.id.text_view) as TextView

2.后期初始化

private lateinit var textView: TextView
…
textView = findViewById(R.id.text_view) as TextView

3. Delegates.notNull()

private var textView: TextView by Delegates.notNull()
…
textView = findViewById(R.id.text_view) as TextView

4.惰性属性

private val textView: TextView by lazy { findViewById(R.id.text_view) as TextView }

5.黄油刀

@BindView(R2.id.text_view)
internal lateinit var textView: TextView

6. Kotterknife

private val textView: TextView by bindView(R.id.text_view)

7. Kotlin Android 扩展

没有代码示例,只需添加正确的导入并开始使用合成生成的属性,如下所示。

import kotlinx.android.synthetic.main.<layout>.*

8.Android数据绑定

binding = FragmentLayoutBinding.inflate(inflater, container, false)
...
binding.textView.text = "Hello"

您可以在Pros/cons of Android view access strategy 中查看每种方法的优缺点

于 2017-11-01T09:52:19.833 回答
4

您可以使用 Kotlin 编写辅助bind函数:

fun <T : View> Activity.bind(@IdRes res : Int) : Lazy<T> {
    @Suppress("UNCHECKED_CAST")    
    return lazy { findViewById(res) as T }
}

然后你可以简单地定义你的变量一次使用val(相当于final在Java中):

class MyActivity : AppCompatActivity() {
    private val button: Button by bind(R.id.button)
}

或者

class MyActivity : AppCompatActivity() {
    private val button by bind<Button>(R.id.button)
}
于 2017-05-29T08:41:56.453 回答