Kotlin之类继承结构

接口

使用interface关键字来声明一个接口

interface Clickable{
    fun click()
}

接下来实现这个接口

class Button : Clickable{
    override fun click() = println("I was clicked")
}

Kotlin中使用冒号代替了Java中的extendsimplements关键字。和Java一样,一个类可以实现多个任意接口,但只能继承一个类

与Java中的@Override注解类似,override修饰符用来标注被重写的父类的接口的方法和属性。

接口可以有一个默认的实现,当然也可以在子类中重新定义showOff函数的实现

interface Clickable{
    fun click()
    fun showOff() = println("I'm clickable!") //带默认实现的方法
}

定义另一个实现了同样方法的接口

interface Focusable{
    fun setFocus(b:Boolean) = println("I ${if (b) "got" else "lose"} focus")
    fun showOff() = println("I'm focusable!")
}

然后在一个子类中同时实现这两个接口,它们每一个都包含了带默认实现的showOff方法,那么子类中你必须要实现这个方法,在Java中可以把基类的名字放在super关键字的前面;但是在Kotlin中需要把基类的名字放在尖括号中

class Button2 : Clickable,Focusable{
override fun showOff() {
    super<Clickable>.showOff()
    super<Focusable>.showOff()
}
override fun click() {}

}

注意:在Java中实现Kotlin的接口,Java并不支持接口中的默认方法,子类必须实现方法体

open、final和abstract修饰符:默认为final

声明一个带openopen

open class RichButton : Clickable{ //这个类是open的:其他类可以继承它
    override fun click() {}//这个函数重写了一个open函数并且它本身同样是open的
    fun disable(){} //这个函数是final的:不能在子类中重写它
    open fun animate(){}//这个函数是open的:可以在子类中重写它
}

注意:如果你重写了一个基类或者接口的成员,重写了的成员同样默认是open的。如果你想阻止你类的子类重写你的实现,可以显示地将重写的成员标注为final

open class RichButton: Clickable {
    final override fun click(){}
}

抽象类

在Kotlin中,与Java一样,可以将一个类声明成abstract的,这种类不能被实例化。并且它内部的抽象成员始终是open的,因此不需要显示地使用open修饰符

abstract class Animated{
    abstract fun animate()
    open fun stopAnimating(){}
    fun animateTwice(){}
}

修饰符

  • 访问修饰符

    普通类中成员默认为final;接口、抽象类中成员默认为open;若子类继承的父类方法为open,那么子类重写后的方法也为open
修饰符 相关成员 评注
final 不能被重写 类中成员默认使用
open 可以被重写 需要明确的表明
abstract 必须被重写 只能在抽象类中使用;抽象成员不能有实现
override 重写父类或接口中的成员 如果没有使用final表明,重写的成员默认是开放的
  • 可见性修饰符(默认为public)
修饰符 类成员 顶层声明
public(默认) 所有地方可见 所有地方可见
internal 模块中可见 模块中可见
protected 子类中可见 ------------
private 类中可见 文件中可见

内部类和嵌套类

Kotlin没有显示的嵌套类与Java中的static嵌套类一样,要把它变成内部类来持有一个外部类的引用要用inner,下面是Java和Kotlin在这个行为上的不同:

类A在另一个类B中声明 在Java中 在Kotlin中
嵌套类(不存储外部类的引用) static class A class A
内部类(存储外部类的引用) class A inner class A

密封类:定义受限的类继承结构

在使用when表达式时,Kotlin编译期会强制检查默认选项。所以不得不添加一个默认的分支。Kotlin为这个问题提供了一个解决方案:sealed类。为父类添加一个sealed修饰符,对可能创建的子类做出严格的限制,所有的直接子类都必须嵌套在父类中

 sealed class Expr  //将基类标记成密封的,sealed修饰符隐含的是open的
{
    class Num(val value:Int) : Expr()   //将所有可能的类作为嵌套类
    class Sum(val left:Expr,val right:Expr) : Expr()
}

fun simplifyEval2(e:Expr):Int =
    when(e){// “when”表达式涵盖了所有可能的情况,所以不再需要“else”分支
       is Expr.Num ->
            e.value
       is Expr.Sum ->
            simplifyEval2(e.left) + simplifyEval2(e.right)
    }
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一朝寒艳醉篱笆,旋染墨馨盈素纱。 梦少长嗟无彩笔,情多但恐累黄花。 羞将浅句颦儿效,敢捧痴心陶令夸。 幽韵自珍霜月...
    逸塵居士阅读 242评论 0 0
  • 咖啡人带您品天下咖啡——今天,广西的咖啡人一定不平凡。因为,以前的咖啡人是自娱自乐、自由自在、自己行动,而今后的咖...
    cb9fa4682249阅读 468评论 0 1
  • 周末,去了一趟方所书店,这是一家集生活居家美物、特色服饰、咖啡小馆、展览讲座空间等多领域为一体的书店,位于广州目前...
    Miss了了阅读 1,375评论 8 5
  • 5018年3月10日星期六天气晴 今天是婆婆妈62周岁的生日,全家起的都很早,尤其是儿子好几天前就嘟囔着要回老家陪...
    一年级一班宋奕辰妈妈阅读 211评论 0 0