-
SealedClass
又称为印章类或密封类,密封类用于子类类型有限的类。 - 密封类与枚举的区别在于,密封类更加在意类型,枚举更加在意数据。
密封类用来表示受限的类继承结构,当一个值为有限的几种类型而不能有任何其他类型。在某种意义上,密封类是枚举类的扩展。枚举类型的值集合是受限的,但每个枚举常量只存在一个 实例,密封类的一个子类可以有包含状态的多个实例。
Kotlin中密封类的出现,在于它定义了一种受限的类继承结构,可以保证编写出更加安全的代码。
由来
//定义接口
interface InterfaceName
//两个类实现接口
class ClassName1:InterfaceName{}
class ClassName2:InterfaceName{}
//通过传入接口对象使用when来判断传入的对象是那种类型。
fun check(i:InterfaceName):String = when(i){
is ClassName1->"1"
is ClassName2->"2"
else -> throw IllegalArgumentException("Unknown class")
}
//测试运行
fun main(args:Array<String>){
val obj = ClassName1()
val result = check(obj)
println(result)//1
}
上面代码存在潜在的问题:加入此时再添加一个子类,此时when
并没有发现子类增加了也不会报错,可能需要给when
增加一个新分支,但由于没有报错可能就会被我们忽略了。基于以上原因,Kotlin为了解决上述潜在问题便提出了密封类。
声明
声明密封类使用关键字sealed
修饰符,需要再类名前添加sealed
修饰符。密封类可以有子类,但所有子类必须要内嵌到密封类中。
sealed class Compute{
class Add(val value:Int):Compute()
class Substract(val value:Int):Compute()
class Multiply(val value:Int):Compute()
class Divide(val value:Int):Compute()
}
sealed class Cmd{
class Play(val url:String, val pos:Long=0):Cmd()
class Seek(val pos:Long):Cmd()
object Pause:Cmd()
object Resume:Cmd()
object Stop:Cmd()
}
使用注意
-
sealed
关键字不能修饰接口interface
和抽象类abstract class
- 密封类时自身抽象的,不能直接实例化但可以有抽象成员。
- 密封类不允许拥有非私有
private
构造函数,因为它的构造函数默认是私有的。 - 密封类封装子类的类(间接继承者 )可以放到任何位置,而无需在同样一个文件中。
密封类的关键好处在于使用when
表达式时,如果能验证语句覆盖了所有情况则无需为该语句再添加一个else
子句。
Kotlin中类不是默认final
的吗?那么密封类为什么可以继承呢?因为被sealed
修饰的类默认是open
的,所以可以被继承,另外无需显式地使用open
修饰符。