当传递null时,有没有办法在非可选参数上使用默认值?

时间:2017-06-23 19:15:26

标签: kotlin nullable default-parameters

例如,如果我有以下数据类:

data class Data(
    val name: String = "",
    val number: Long = 0
)

可以返回null的函数:

fun newName(): String? {}

fun newNumber(): Long? {}

我知道如果不是null,我可以使用以下函数来使用函数的值:

val newName = newName()
val newNumber = newNumber()

val data = Data(
        if (newName != null) newName else "",
        if (newNumber != null) newNumber else 0
)

但是当值为Data时,有没有办法只使用null类的构造函数中指定的默认值?

我在文档中找不到任何内容,但我希望这样的内容能够起作用:

val data = Data(newName()?, newNumber()?)

但那不会编译。

2 个答案:

答案 0 :(得分:21)

您可以在接收null时创建使用相同默认值的secondary constructor

data class Data(
        val name: String = "",
        val number: Long = 0
) {
    constructor(
            name: String? = null,
            number: Long? = null
    ) : this(
            name ?: "",
            number ?: 0
    )
}

答案 1 :(得分:11)

辅助构造函数仅支持Nullable原始属性。这意味着如果属性不是基本类型,它将产生2个相同的构造函数,例如:

data class Data(val name: String) {
    constructor(name: String? = null) : this(name ?: "foo");
    // ^--- report constructor signature error                
}

data class Data(val number: Long = 0) {
     constructor(number: Long? = null) : this(number ?: 0)
     //                  ^--- No problem since there are 2 constructors generated:
     //                       Data(long number) and Data(java.lang.Long number)
}

另一种方法是使用invoke运算符,例如:

data class Data(val name: String) {
    companion object {
        operator fun invoke(name: String? = null) = Data(name ?: "")
    }
}

IF 该类不是数据类,那么您可以从参数中延迟初始化属性,而不是在主构造函数上定义属性,例如:

class Data(name: String? = null, number: Long? = null) {
    val name = name ?: ""
    val number = number ?: 0
}