3.14.1. Data class #
Kotlin can create a class that contains only data with the keyword data :
data class User(val name: String, val age: Int) The compiler automatically extracts the following functions from the main constructor based on all declared attributes:
equals()/hashCode()toString()format such as"User(name=John, age=42)"componentN() functionscorresponding to attributes, in order of declarationcopy()function
If these functions are explicitly defined in the class or inherited from thesuperclass, they will no longer be generated.
In order to ensure that the generated code is consistent and meaningful, thedata class needs to meet the following conditions:
The main constructor contains at least one parameter.
All parameters of the main constructor must be identified as
valorvar;Data classes cannot be declared as
abstract,open,sealed, orinner;Data classes cannot inherit from other classes (but can implement interfaces).
3.14.2. Copy #
Copy and use copy() function, which we can use to copy the object and modify some properties. For the User class above, the implementation will besimilar to the following:
fun copy(name: String = this.name, age: Int = this.age) = User(name, age) 3.14.3. Example #
Use copy class replication User data class, and modify age attributes:
data class User(val name: String, val age: Int) fun main(args: Array<String>) { val jack = User(name = "Jack", age = 1) val olderJack = jack.copy(age = 2) println(jack) println(olderJack) } The output is as follows:
User(name=Jack, age=1) User(name=Jack, age=2) 3.14.4. Data classes and deconstructing declarations #
Component functions allow data classes to be used in deconstructing declarations:
val jane = User("Jane", 35) val (name, age) = jane println("$name, $age years of age") // prints "Jane, 35 years of age" 3.14.5. Standard data class #
The standard library provides Pair and Triple . In most cases, named data classes are a better design choice because the code is more readable and provides meaningful names and attributes.
3.14.6. Sealed class #
Sealed classes are used to represent restricted class inheritance structures: when a value is limited to a limited number of types and there cannot be any other types. In a sense, they are extensions of enumerated classes: the set of values of enumerated types is also limited, but there isonly one instance of each enumeration constant, and a subclass of a sealed class can have multiple instances of states that can be contained.
Declare a sealed class, using the sealed a modified class, a sealed class can have subclasses, but all subclasses must be embedded in the sealedclass.
sealed cannot modify interface , abstract class ( warning will be reported, but compilation errors will not occur)
sealed class Expr data class Const(val number: Double) : Expr() data class Sum(val e1: Expr, val e2: Expr) : Expr() object NotANumber : Expr() fun eval(expr: Expr): Double = when (expr) { is Const -> expr.number is Sum -> eval(expr.e1) + eval(expr.e2) NotANumber -> Double.NaN } The key benefit of using a sealed class is to use the when in the case of an expression, if you can verify that the statement covers all cases, youdo not need to add another else clause.
fun eval(expr: Expr): Double = when(expr) { is Expr.Const -> expr.number is Expr.Sum -> eval(expr.e1) + eval(expr.e2) Expr.NotANumber -> Double.NaN // The `else` clause is no longer needed because we have covered all cases }