密封类用来表示受限的类继承结构:当一个值为有限几种的类型、而不能有任何其他类型时。
说白了就是固定的类继承结构,类似于枚举enum,但是enum只能有一个类型的实例对象。
enum class Day {
MON,
TEU,
WED,
THU,
FRI,
SAT,
SUN
}
要声明一个密封类,需要在类名前面添加 sealed
修饰符。虽然密封类也可以有子类,但是所有子类都必须在与密封类自身相同的文件中声明。(在 Kotlin 1.1 之前, 该规则更加严格:子类必须嵌套在密封类声明的内部)。
sealed class Expr
data class Const(val number: Double) : Expr()
data class Sum(val e1: Expr, val e2: Expr) : Expr()
object NotANumber : Expr()
特点:
-
sealed
修饰符 - 密封类也可以有子类,但是所有子类都必须在与密封类自身相同的文件中声明
- 密封类的一个子类可以有可包含状态的多个实例。
- 不允许有非-private 构造函数
- 一个密封类是自身抽象的,它不能直接实例化但可以有抽象(abstract)成员。
- 扩展密封类子类的类(间接继承者)可以放在任何位置,而无需在同一个文件中。
密封类还有一个用处,就是 when
表达式 的时候,如果能够验证语句覆盖了所有情况,就不需要为该语句再添加一个 else
子句了。当然,这只有当你用 when
作为表达式(使用结果)而不是作为语句时才有用。
fun eval(expr: Expr): Double = when(expr) {
is Const -> expr.number
is Sum -> eval(expr.e1) + eval(expr.e2)
NotANumber -> Double.NaN
// 不再需要 `else` 子句,因为我们已经覆盖了所有的情况
}