Kotlin数据类可以有多个构造函数吗?

问题描述 投票:20回答:6

我知道数据类就像kotlin中的简单模型一样,默认情况下使用getter和setter,就像这样简单:

data class User(val name: String, val age: Int)

是否可以为该数据类声明第二个构造函数?

constructor kotlin
6个回答
18
投票

是的,但是应该初始化每个变量,因此您可以在数据类构造函数中设置默认参数,如下所示:

data class Person(val age: Int, val name: String = "Person without name")

现在,您可以通过两种方式创建此数据类的实例

  • Person(30)
  • Person(20, "Bob")

18
投票

Kotlin数据类必须具有定义至少一个成员的主构造函数。除此之外,您可以添加辅助构造函数,如Classes and Inheritance - Secondary Constructors中所述。

对于您的类和示例辅助构造函数:

data class User(val name: String, val age: Int) {
    constructor(name: String): this(name, -1) { ... }
}

请注意,辅助构造函数必须在其定义中委托给主构造函数。

虽然辅助构造函数的许多共同点可以通过使用参数的默认值来解决。在上面的例子中,您可以简化为:

data class User(val name: String, val age: Int = -1) 

如果从Java调用这些文件,你应该阅读有关如何生成重载的Java interop - Java calling Kotlin文档,有时可能还有其他特殊情况下的NoArg Compiler Plugin文档。


4
投票

更新了数据类的答案:

是的,你可以,但你需要将所有内容委托给主要构造函数

data class User(val name: String, val age: Int)
{
    constructor(name: String): this(name, -1) {
    }

    constructor(age: Int): this("Anon", age) {
    }
}

// Anon name: Anon
println("Anon name: " + User(30).name)

// No age: -1
println("No age: " + User("Name").age)

// Name: Name age: 20
val u = User("Name", 20)
println("Name: " + u.name + " age: " + u.age)

您也可以像Alexey那样在主构造函数中设置默认值。


2
投票

是的,我们可以使用类似下面的代码,并且在数据类的主构造函数中应该有min一个参数。

data class SampleData(val name: String, val age: Int) {
    constructor(name: String, age: Int, email: String) : this(name, age) {

    }
}

1
投票

主构造函数中的默认值消除了对辅助构造函数的许多需求,但如果所需实例依赖于基于必须分析的数据的逻辑,则更好的答案可能是使用伴随对象。

data class KeyTag(val a: String, val b: Int, val c: Double) {
    companion object Factory {
        val empty = KeyTag("", 0, 0.0)

        fun create(bigString: String): KeyTag {
            // Logic to extract appropriate values for arguments a, b, c
            return KeyTag(a, b, c)
        }

        fun bake(i: Int): KeyTag = KeyTag("$i", i, i.toDouble())
    }
}

用法是:

val ks = KeyTag.create("abc:1:10.0")
val ke = KeyTag.empty
val kb = KeyTag.bake(2)

0
投票

数据类将确保一致性和有意义的行为,我们还需要具有不可变性的val。

data class SampleData(val name: String, val age: Int, val email: String ?= null) {
constructor(name: String, age: Int) : this(name, age, null) {

}

}

辅助构造函数必须在其定义中委托给主构造函数,因此为了保持不变性,使用“null”将起作用。


0
投票

指示Kotlin编译器为此函数生成替换默认参数值的重载。如果一个方法有N个参数且其中M个有默认值,则会产生M个重载:第一个采用N-1个参数(除了最后一个采用默认值),第二个采用N-2个参数,所以上。

data class User @JvmOverloads  constructor(
var email: String="",
var password: String=""

)

© www.soinside.com 2019 - 2024. All rights reserved.