Kotlin密封类

  • 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){
    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修饰符。

你可能感兴趣的:(Kotlin密封类)