Kotlin学习之继承
@(Kotlin学习)
Kotlin中的继承模式与Java中存在差异:
- Kotlin中所有类的超类是Any,而不是Object;
- Kotlin中非抽象类默认不可继承;
- Kotlin中非抽象类函数和类属性默认不可覆盖。
一、open关键字
在Kotlin中所有类都有一个共同的超类Any,对于没有超类型声明的类是默认超类:
class Person
要声明一个显式的超类型,需要把类型放到类头的冒号之后:
open class Base(p: Int)
class Derived(p: Int) : Base(p)
- 如果该类有一个主构造函数,其基类型可以用主构造函数参数就地初始化。
- 如果类没有主构造函数,那么每个次构造函数必须使用super关键字初始化其基类型,或者委托给另一个构造函数做。
class MyView : View {
constructor(ctx: Context) : super(ctx)
constructor(ctx: Context, attrs: AttributeSet) : super(ctx, attrs)
}
open关键字在Kotlin中可以用在定义非抽象的类、类函数和类属性之前,用来将它们标记为可继承的:
默认情况下,在Kotlin中所有的类都是不能覆盖的。
二、覆盖方法
Kotlin力求清晰显示,与Java不同,Kotlin需要显示标注可覆盖的成员和覆盖后的成员;
open class Base{
open fun v(){}
fun nv(){}
}
class Dervied():Base(){
override fun v() {}
}
注意:
- 如果函数没有标注open,则子类中不允许定义相同名字的函数,无论加不加override;
- 在一个没有用open标注的类中,开放成员是禁止的。
- 标记为override的成员本身是开放的,它可以在子类中覆盖。如果想要禁止再次覆盖,可以使用final关键字:
open class AnotherDerived() : Base() {
final override fun v() {}
}
三、覆盖属性
在超类中声明然后在派生类中重新声明的属性必须以override开头,并且必须具有兼容的类型。
每个声明的属性可以由具有初始化器的属性或者具有getter方法的属性覆盖。
open class Foo {
open val x: Int get() { …… }
}
class Bar1 : Foo() {
override val x: Int = ……
}
- 可以用一个var属性覆盖一个val属性,反之则不行;
- 一个val属性本质上声明了一个getter方法;将其覆盖为var只是在子类中额外声明一个setter方法。
- 可以在主构造函数中定义要覆盖的属性。
可以在主构造函数中使用override关键字作为属性声明的一部分:
interface Foo{
val count:Int
}
class Bar1(override val count:Int):Foo
class Bar2:Foo{
override var count:Int=0
}
四、覆盖规则
在Kotlin中,实现继承由下属规则规定:如果一个类从它的直接超类继承相同成员的多个实现,必须覆盖这个成员并提供其自己的实现。
为了表示采用从哪个超类型继承的实现,使用由尖括号中超类型名限定的super:
open class A {
open fun f() {
print("A")
}
fun a() {
print("a")
}
}
interface B {
fun f() {
print("B")
}
fun b() {
print("b")
}
}
class C():A(),B{
override fun f() {
super<A>.f()
super<B>.f()
}
}
五、继承类的写法和构造方法
在Kotlin中没有extends关键字,声明一个类继承自另一个标记为open的类的方法是:
class 子类[(主构造方法参数)] : 父类[(主构造方法参数)]{......}
- 冒号(:):在Kotlin中表示前者属于后者类型;
在Kotlin中,如果父类定义了主构造函数,子类就必须显式地调用父类的主构造函数。
网友评论